mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-24 23:38:56 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			255 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div class="w-full h-full relative">
 | |
|     <form class="w-full h-full px-4 py-6" @submit.prevent="submitForm">
 | |
|       <div class="flex -mx-1">
 | |
|         <div class="w-1/2 px-1">
 | |
|           <ui-text-input-with-label ref="titleInput" v-model="details.title" :label="$strings.LabelTitle" trim-whitespace @input="handleInputChange" />
 | |
|         </div>
 | |
|         <div class="grow px-1">
 | |
|           <ui-text-input-with-label ref="authorInput" v-model="details.author" :label="$strings.LabelAuthor" trim-whitespace @input="handleInputChange" />
 | |
|         </div>
 | |
|       </div>
 | |
| 
 | |
|       <ui-text-input-with-label ref="feedUrlInput" v-model="details.feedUrl" :label="$strings.LabelRSSFeedURL" trim-whitespace class="mt-2" @input="handleInputChange" />
 | |
| 
 | |
|       <ui-textarea-with-label ref="descriptionInput" v-model="details.description" :rows="3" :label="$strings.LabelDescription" class="mt-2" @input="handleInputChange" />
 | |
| 
 | |
|       <div class="flex mt-2 -mx-1">
 | |
|         <div class="w-1/2 px-1">
 | |
|           <ui-multi-select ref="genresSelect" v-model="details.genres" :label="$strings.LabelGenres" :items="genres" @input="handleInputChange" />
 | |
|         </div>
 | |
|         <div class="grow px-1">
 | |
|           <ui-multi-select ref="tagsSelect" v-model="newTags" :label="$strings.LabelTags" :items="tags" @input="handleInputChange" />
 | |
|         </div>
 | |
|       </div>
 | |
| 
 | |
|       <div class="flex mt-2 -mx-1">
 | |
|         <div class="w-1/4 px-1">
 | |
|           <ui-text-input-with-label ref="releaseDateInput" v-model="details.releaseDate" :label="$strings.LabelReleaseDate" trim-whitespace @input="handleInputChange" />
 | |
|         </div>
 | |
|         <div class="w-1/4 px-1">
 | |
|           <ui-text-input-with-label ref="itunesIdInput" v-model="details.itunesId" label="iTunes ID" trim-whitespace @input="handleInputChange" />
 | |
|         </div>
 | |
|         <div class="w-1/4 px-1">
 | |
|           <ui-text-input-with-label ref="languageInput" v-model="details.language" :label="$strings.LabelLanguage" trim-whitespace @input="handleInputChange" />
 | |
|         </div>
 | |
|         <div class="grow px-1 pt-6">
 | |
|           <div class="flex justify-center">
 | |
|             <ui-checkbox v-model="details.explicit" :label="$strings.LabelExplicit" checkbox-bg="primary" border-color="gray-600" label-class="pl-2 text-base font-semibold" @input="handleInputChange" />
 | |
|           </div>
 | |
|         </div>
 | |
|       </div>
 | |
|       <div class="flex mt-2 -mx-1">
 | |
|         <div class="w-1/4 px-1">
 | |
|           <ui-dropdown :label="$strings.LabelPodcastType" v-model="details.type" :items="podcastTypes" small class="max-w-52" @input="handleInputChange" />
 | |
|         </div>
 | |
|       </div>
 | |
|     </form>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| export default {
 | |
|   props: {
 | |
|     libraryItem: {
 | |
|       type: Object,
 | |
|       default: () => {}
 | |
|     }
 | |
|   },
 | |
|   data() {
 | |
|     return {
 | |
|       details: {
 | |
|         title: null,
 | |
|         author: null,
 | |
|         description: null,
 | |
|         releaseDate: null,
 | |
|         genres: [],
 | |
|         feedUrl: null,
 | |
|         imageUrl: null,
 | |
|         itunesPageUrl: null,
 | |
|         itunesId: null,
 | |
|         itunesArtistId: null,
 | |
|         explicit: false,
 | |
|         language: null,
 | |
|         type: null
 | |
|       },
 | |
|       newTags: []
 | |
|     }
 | |
|   },
 | |
|   watch: {
 | |
|     libraryItem: {
 | |
|       immediate: true,
 | |
|       handler(newVal) {
 | |
|         if (newVal) this.init()
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   computed: {
 | |
|     media() {
 | |
|       return this.libraryItem ? this.libraryItem.media || {} : {}
 | |
|     },
 | |
|     mediaMetadata() {
 | |
|       return this.media.metadata || {}
 | |
|     },
 | |
|     genres() {
 | |
|       return this.filterData.genres || []
 | |
|     },
 | |
|     tags() {
 | |
|       return this.filterData.tags || []
 | |
|     },
 | |
|     filterData() {
 | |
|       return this.$store.state.libraries.filterData || {}
 | |
|     },
 | |
|     podcastTypes() {
 | |
|       return this.$store.state.globals.podcastTypes.map((e) => {
 | |
|         return {
 | |
|           text: this.$strings[e.descriptionKey] || e.text,
 | |
|           value: e.value
 | |
|         }
 | |
|       })
 | |
|     }
 | |
|   },
 | |
|   methods: {
 | |
|     handleInputChange() {
 | |
|       this.$emit('change', {
 | |
|         libraryItemId: this.libraryItem.id,
 | |
|         hasChanges: this.checkForChanges().hasChanges
 | |
|       })
 | |
|     },
 | |
|     getDetails() {
 | |
|       this.forceBlur()
 | |
|       return this.checkForChanges()
 | |
|     },
 | |
|     getTitleAndAuthorName() {
 | |
|       this.forceBlur()
 | |
|       return {
 | |
|         title: this.details.title,
 | |
|         author: this.details.author
 | |
|       }
 | |
|     },
 | |
|     mapBatchDetails(batchDetails, mapType = 'overwrite') {
 | |
|       for (const key in batchDetails) {
 | |
|         if (mapType === 'append') {
 | |
|           if (key === 'tags') {
 | |
|             // Concat and remove dupes
 | |
|             this.newTags = [...new Set(this.newTags.concat(batchDetails.tags))]
 | |
|           } else if (key === 'genres') {
 | |
|             // Concat and remove dupes
 | |
|             this.details[key] = [...new Set(this.details[key].concat(batchDetails[key]))]
 | |
|           }
 | |
|         } else {
 | |
|           if (key === 'tags') {
 | |
|             this.newTags = [...batchDetails.tags]
 | |
|           } else if (key === 'genres') {
 | |
|             this.details[key] = [...batchDetails[key]]
 | |
|           } else {
 | |
|             this.details[key] = batchDetails[key]
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       this.handleInputChange()
 | |
|     },
 | |
|     forceBlur() {
 | |
|       if (this.$refs.titleInput) this.$refs.titleInput.blur()
 | |
|       if (this.$refs.authorInput) this.$refs.authorInput.blur()
 | |
|       if (this.$refs.releaseDateInput) this.$refs.releaseDateInput.blur()
 | |
|       if (this.$refs.descriptionInput) this.$refs.descriptionInput.blur()
 | |
|       if (this.$refs.feedUrlInput) this.$refs.feedUrlInput.blur()
 | |
|       if (this.$refs.itunesIdInput) this.$refs.itunesIdInput.blur()
 | |
|       if (this.$refs.languageInput) this.$refs.languageInput.blur()
 | |
| 
 | |
|       if (this.$refs.genresSelect && this.$refs.genresSelect.isFocused) {
 | |
|         this.$refs.genresSelect.forceBlur()
 | |
|       }
 | |
|       if (this.$refs.tagsSelect && this.$refs.tagsSelect.isFocused) {
 | |
|         this.$refs.tagsSelect.forceBlur()
 | |
|       }
 | |
|     },
 | |
|     stringArrayEqual(array1, array2) {
 | |
|       // return false if different
 | |
|       if (array1.length !== array2.length) return false
 | |
|       for (var item of array1) {
 | |
|         if (!array2.includes(item)) return false
 | |
|       }
 | |
|       return true
 | |
|     },
 | |
|     objectArrayEqual(array1, array2) {
 | |
|       const isIterable = (value) => {
 | |
|         return Symbol.iterator in Object(value)
 | |
|       }
 | |
|       if (!isIterable(array1) || !isIterable(array2)) {
 | |
|         console.error(array1, array2)
 | |
|         throw new Error('Invalid arrays passed in')
 | |
|       }
 | |
| 
 | |
|       // array of objects with id key
 | |
|       if (array1.length !== array2.length) return false
 | |
| 
 | |
|       for (var item of array1) {
 | |
|         var matchingItem = array2.find((a) => a.id === item.id)
 | |
|         if (!matchingItem) return false
 | |
|         for (var key in item) {
 | |
|           if (item[key] !== matchingItem[key]) {
 | |
|             // console.log('Object array item keys changed', key, item[key], matchingItem[key])
 | |
|             return false
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       return true
 | |
|     },
 | |
|     checkForChanges() {
 | |
|       var metadata = {}
 | |
|       for (const key in this.details) {
 | |
|         var newValue = this.details[key]
 | |
|         var oldValue = this.mediaMetadata[key]
 | |
|         // Key cleared out or key first populated
 | |
|         if ((!newValue && oldValue) || (newValue && !oldValue)) {
 | |
|           metadata[key] = newValue
 | |
|         } else if (key === 'genres') {
 | |
|           // Check array of strings
 | |
|           if (!this.stringArrayEqual(newValue, oldValue)) {
 | |
|             metadata[key] = [...newValue]
 | |
|           }
 | |
|         } else if (newValue && newValue != oldValue) {
 | |
|           // Intentional !=
 | |
|           metadata[key] = newValue
 | |
|         }
 | |
|       }
 | |
|       var updatePayload = {}
 | |
|       if (!!Object.keys(metadata).length) updatePayload.metadata = metadata
 | |
| 
 | |
|       if (!this.stringArrayEqual(this.newTags, this.media.tags || [])) {
 | |
|         updatePayload.tags = [...this.newTags]
 | |
|       }
 | |
| 
 | |
|       return {
 | |
|         updatePayload,
 | |
|         hasChanges: !!Object.keys(updatePayload).length
 | |
|       }
 | |
|     },
 | |
|     init() {
 | |
|       this.details.title = this.mediaMetadata.title
 | |
|       this.details.author = this.mediaMetadata.author || ''
 | |
|       this.details.description = this.mediaMetadata.description || ''
 | |
|       this.details.releaseDate = this.mediaMetadata.releaseDate || ''
 | |
|       this.details.genres = [...(this.mediaMetadata.genres || [])]
 | |
|       this.details.feedUrl = this.mediaMetadata.feedUrl || ''
 | |
|       this.details.imageUrl = this.mediaMetadata.imageUrl || ''
 | |
|       this.details.itunesPageUrl = this.mediaMetadata.itunesPageUrl || ''
 | |
|       this.details.itunesId = this.mediaMetadata.itunesId || ''
 | |
|       this.details.itunesArtistId = this.mediaMetadata.itunesArtistId || ''
 | |
|       this.details.language = this.mediaMetadata.language || ''
 | |
|       this.details.explicit = !!this.mediaMetadata.explicit
 | |
|       this.details.type = this.mediaMetadata.type || 'episodic'
 | |
| 
 | |
|       this.newTags = [...(this.media.tags || [])]
 | |
|     },
 | |
|     submitForm() {
 | |
|       this.$emit('submit')
 | |
|     }
 | |
|   },
 | |
|   mounted() {}
 | |
| }
 | |
| </script>
 |