mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-26 16:22:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			249 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			249 lines
		
	
	
		
			8.8 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="font-book 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-user-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.selectedLibraryItems || []
 | |
|     },
 | |
|     currentLibraryId() {
 | |
|       return this.$store.state.libraries.currentLibraryId
 | |
|     }
 | |
|   },
 | |
|   methods: {
 | |
|     loadCollections() {
 | |
|       if (!this.collections.length) {
 | |
|         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('Failed to load collections')
 | |
|           })
 | |
|           .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>
 | |
| 
 | |
| <style>
 | |
| .list-complete-item {
 | |
|   transition: all 0.8s ease;
 | |
| }
 | |
| 
 | |
| .list-complete-enter-from,
 | |
| .list-complete-leave-to {
 | |
|   opacity: 0;
 | |
|   transform: translateY(30px);
 | |
| }
 | |
| 
 | |
| .list-complete-leave-active {
 | |
|   position: absolute;
 | |
| }
 | |
| </style> |