mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-27 11:18:14 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			232 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			232 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<template>
 | 
						|
  <modals-modal v-model="show" name="collections" :processing="processing" :width="500" :height="'unset'">
 | 
						|
    <template #outer>
 | 
						|
      <div class="absolute top-0 left-0 p-5 w-2/3 overflow-hidden pointer-events-none">
 | 
						|
        <p class="text-3xl text-white truncate">{{ title }}</p>
 | 
						|
      </div>
 | 
						|
    </template>
 | 
						|
 | 
						|
    <div ref="container" class="w-full rounded-lg bg-primary box-shadow-md overflow-y-auto overflow-x-hidden" style="max-height: 80vh">
 | 
						|
      <div v-if="show" class="w-full h-full">
 | 
						|
        <div class="py-4 px-4">
 | 
						|
          <h1 v-if="!showBatchCollectionModal" class="text-2xl">{{ $strings.LabelAddToCollection }}</h1>
 | 
						|
          <h1 v-else class="text-2xl">{{ $getString('LabelAddToCollectionBatch', [selectedBookIds.length]) }}</h1>
 | 
						|
        </div>
 | 
						|
        <div class="w-full overflow-y-auto overflow-x-hidden max-h-96">
 | 
						|
          <transition-group name="list-complete" tag="div">
 | 
						|
            <template v-for="collection in sortedCollections">
 | 
						|
              <modals-collections-collection-item :key="collection.id" :collection="collection" :book-cover-aspect-ratio="bookCoverAspectRatio" class="list-complete-item" @add="addToCollection" @remove="removeFromCollection" @close="show = false" />
 | 
						|
            </template>
 | 
						|
          </transition-group>
 | 
						|
        </div>
 | 
						|
        <div v-if="!collections.length" class="flex h-32 items-center justify-center">
 | 
						|
          <p class="text-xl">{{ $strings.MessageNoCollections }}</p>
 | 
						|
        </div>
 | 
						|
        <div class="w-full h-px bg-white bg-opacity-10" />
 | 
						|
        <form @submit.prevent="submitCreateCollection">
 | 
						|
          <div class="flex px-4 py-2 items-center text-center border-b border-white border-opacity-10 text-white text-opacity-80">
 | 
						|
            <div class="flex-grow px-2">
 | 
						|
              <ui-text-input v-model="newCollectionName" :placeholder="$strings.PlaceholderNewCollection" class="w-full" />
 | 
						|
            </div>
 | 
						|
            <ui-btn type="submit" color="success" :padding-x="4" class="h-10">{{ $strings.ButtonCreate }}</ui-btn>
 | 
						|
          </div>
 | 
						|
        </form>
 | 
						|
      </div>
 | 
						|
    </div>
 | 
						|
  </modals-modal>
 | 
						|
</template>
 | 
						|
 | 
						|
<script>
 | 
						|
export default {
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      newCollectionName: '',
 | 
						|
      processing: false
 | 
						|
    }
 | 
						|
  },
 | 
						|
  watch: {
 | 
						|
    show(newVal) {
 | 
						|
      if (newVal) {
 | 
						|
        this.loadCollections()
 | 
						|
        this.newCollectionName = ''
 | 
						|
      } else {
 | 
						|
        this.$store.commit('setSelectedLibraryItem', null)
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    show: {
 | 
						|
      get() {
 | 
						|
        return this.$store.state.globals.showCollectionsModal
 | 
						|
      },
 | 
						|
      set(val) {
 | 
						|
        this.$store.commit('globals/setShowCollectionsModal', val)
 | 
						|
      }
 | 
						|
    },
 | 
						|
    title() {
 | 
						|
      if (this.showBatchCollectionModal) {
 | 
						|
        return this.$getString('MessageItemsSelected', [this.selectedBookIds.length])
 | 
						|
      }
 | 
						|
      return this.selectedLibraryItem ? this.selectedLibraryItem.media.metadata.title : ''
 | 
						|
    },
 | 
						|
    collections() {
 | 
						|
      return this.$store.state.libraries.collections || []
 | 
						|
    },
 | 
						|
    bookCoverAspectRatio() {
 | 
						|
      return this.$store.getters['libraries/getBookCoverAspectRatio']
 | 
						|
    },
 | 
						|
    selectedLibraryItem() {
 | 
						|
      return this.$store.state.selectedLibraryItem
 | 
						|
    },
 | 
						|
    selectedLibraryItemId() {
 | 
						|
      return this.selectedLibraryItem ? this.selectedLibraryItem.id : null
 | 
						|
    },
 | 
						|
    sortedCollections() {
 | 
						|
      return this.collections
 | 
						|
        .map((c) => {
 | 
						|
          var includesBook = false
 | 
						|
          if (this.showBatchCollectionModal) {
 | 
						|
            // Only show collection added if all books are in the collection
 | 
						|
            var collectionBookIds = c.books.map((b) => b.id)
 | 
						|
            includesBook = !this.selectedBookIds.find((id) => !collectionBookIds.includes(id))
 | 
						|
          } else {
 | 
						|
            includesBook = !!c.books.find((b) => b.id === this.selectedLibraryItemId)
 | 
						|
          }
 | 
						|
 | 
						|
          return {
 | 
						|
            isBookIncluded: includesBook,
 | 
						|
            ...c
 | 
						|
          }
 | 
						|
        })
 | 
						|
        .sort((a, b) => (a.isBookIncluded ? -1 : 1))
 | 
						|
    },
 | 
						|
    showBatchCollectionModal() {
 | 
						|
      return this.$store.state.globals.showBatchCollectionModal
 | 
						|
    },
 | 
						|
    selectedBookIds() {
 | 
						|
      return (this.$store.state.globals.selectedMediaItems || []).map((i) => i.id)
 | 
						|
    },
 | 
						|
    currentLibraryId() {
 | 
						|
      return this.$store.state.libraries.currentLibraryId
 | 
						|
    }
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    loadCollections() {
 | 
						|
      this.processing = true
 | 
						|
      this.$axios
 | 
						|
        .$get(`/api/libraries/${this.currentLibraryId}/collections`)
 | 
						|
        .then((data) => {
 | 
						|
          if (data.results) {
 | 
						|
            this.$store.commit('libraries/setCollections', data.results || [])
 | 
						|
          }
 | 
						|
        })
 | 
						|
        .catch((error) => {
 | 
						|
          console.error('Failed to get collections', error)
 | 
						|
          this.$toast.error(this.$strings.ToastFailedToLoadData)
 | 
						|
        })
 | 
						|
        .finally(() => {
 | 
						|
          this.processing = false
 | 
						|
        })
 | 
						|
    },
 | 
						|
    removeFromCollection(collection) {
 | 
						|
      if (!this.selectedLibraryItemId && !this.selectedBookIds.length) return
 | 
						|
      this.processing = true
 | 
						|
 | 
						|
      if (this.showBatchCollectionModal) {
 | 
						|
        // BATCH Remove books
 | 
						|
        this.$axios
 | 
						|
          .$post(`/api/collections/${collection.id}/batch/remove`, { books: this.selectedBookIds })
 | 
						|
          .then((updatedCollection) => {
 | 
						|
            console.log(`Books removed from collection`, updatedCollection)
 | 
						|
            this.$toast.success(this.$strings.ToastCollectionItemsRemoveSuccess)
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
          .catch((error) => {
 | 
						|
            console.error('Failed to remove books from collection', error)
 | 
						|
            this.$toast.error(this.$strings.ToastCollectionItemsRemoveFailed)
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
      } else {
 | 
						|
        // Remove single book
 | 
						|
        this.$axios
 | 
						|
          .$delete(`/api/collections/${collection.id}/book/${this.selectedLibraryItemId}`)
 | 
						|
          .then((updatedCollection) => {
 | 
						|
            console.log(`Book removed from collection`, updatedCollection)
 | 
						|
            this.$toast.success(this.$strings.ToastCollectionItemsRemoveSuccess)
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
          .catch((error) => {
 | 
						|
            console.error('Failed to remove book from collection', error)
 | 
						|
            this.$toast.error(this.$strings.ToastCollectionItemsRemoveFailed)
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
      }
 | 
						|
    },
 | 
						|
    addToCollection(collection) {
 | 
						|
      if (!this.selectedLibraryItemId && !this.selectedBookIds.length) return
 | 
						|
      this.processing = true
 | 
						|
 | 
						|
      if (this.showBatchCollectionModal) {
 | 
						|
        // BATCH Remove books
 | 
						|
        this.$axios
 | 
						|
          .$post(`/api/collections/${collection.id}/batch/add`, { books: this.selectedBookIds })
 | 
						|
          .then((updatedCollection) => {
 | 
						|
            console.log(`Books added to collection`, updatedCollection)
 | 
						|
            this.$toast.success('Books added to collection')
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
          .catch((error) => {
 | 
						|
            console.error('Failed to add books to collection', error)
 | 
						|
            this.$toast.error('Failed to add books to collection')
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
      } else {
 | 
						|
        if (!this.selectedLibraryItemId) return
 | 
						|
 | 
						|
        this.$axios
 | 
						|
          .$post(`/api/collections/${collection.id}/book`, { id: this.selectedLibraryItemId })
 | 
						|
          .then((updatedCollection) => {
 | 
						|
            console.log(`Book added to collection`, updatedCollection)
 | 
						|
            this.$toast.success('Book added to collection')
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
          .catch((error) => {
 | 
						|
            console.error('Failed to add book to collection', error)
 | 
						|
            this.$toast.error('Failed to add book to collection')
 | 
						|
            this.processing = false
 | 
						|
          })
 | 
						|
      }
 | 
						|
    },
 | 
						|
    submitCreateCollection() {
 | 
						|
      if (!this.newCollectionName || (!this.selectedLibraryItemId && !this.selectedBookIds.length)) {
 | 
						|
        return
 | 
						|
      }
 | 
						|
      this.processing = true
 | 
						|
 | 
						|
      var books = this.showBatchCollectionModal ? this.selectedBookIds : [this.selectedLibraryItemId]
 | 
						|
      var newCollection = {
 | 
						|
        books: books,
 | 
						|
        libraryId: this.currentLibraryId,
 | 
						|
        name: this.newCollectionName
 | 
						|
      }
 | 
						|
 | 
						|
      this.$axios
 | 
						|
        .$post('/api/collections', newCollection)
 | 
						|
        .then((data) => {
 | 
						|
          console.log('New Collection Created', data)
 | 
						|
          this.$toast.success(`Collection "${data.name}" created`)
 | 
						|
          this.processing = false
 | 
						|
          this.newCollectionName = ''
 | 
						|
        })
 | 
						|
        .catch((error) => {
 | 
						|
          console.error('Failed to create collection', error)
 | 
						|
          var errMsg = error.response ? error.response.data || '' : ''
 | 
						|
          this.$toast.error(`Failed to create collection: ${errMsg}`)
 | 
						|
          this.processing = false
 | 
						|
        })
 | 
						|
    }
 | 
						|
  },
 | 
						|
  mounted() {}
 | 
						|
}
 | 
						|
</script>
 |