mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-26 16:22:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			167 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div ref="wrapper" role="dialog" aria-modal="true" class="modal modal-bg w-full h-full fixed top-0 left-0 bg-primary items-center justify-center opacity-0 hidden" :class="`z-${zIndex} bg-primary/${bgOpacity}`">
 | |
|     <div class="absolute top-0 left-0 right-0 w-full h-36 bg-linear-to-t from-transparent via-black-500 to-black-700 opacity-90 pointer-events-none" />
 | |
| 
 | |
|     <button class="absolute top-4 right-4 landscape:top-4 landscape:right-4 md:portrait:top-5 md:portrait:right-5 lg:top-5 lg:right-5 inline-flex text-gray-200 hover:text-white" aria-label="Close modal" @click="clickClose">
 | |
|       <span class="material-symbols text-2xl landscape:text-2xl md:portrait:text-4xl lg:text-4xl">close</span>
 | |
|     </button>
 | |
|     <slot name="outer" />
 | |
|     <div ref="content" tabindex="0" style="min-width: 380px; min-height: 200px; max-width: 100vw" class="relative text-white outline-hidden" :style="{ height: modalHeight, width: modalWidth, marginTop: contentMarginTop + 'px' }" @mousedown="mousedownModal" @mouseup="mouseupModal" v-click-outside="clickBg">
 | |
|       <slot />
 | |
|       <div v-if="processing" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full bg-black/60 rounded-lg flex items-center justify-center">
 | |
|         <ui-loading-indicator />
 | |
|       </div>
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| export default {
 | |
|   props: {
 | |
|     name: String,
 | |
|     value: Boolean,
 | |
|     processing: Boolean,
 | |
|     persistent: {
 | |
|       type: Boolean,
 | |
|       default: true
 | |
|     },
 | |
|     width: {
 | |
|       type: [String, Number],
 | |
|       default: 500
 | |
|     },
 | |
|     height: {
 | |
|       type: [String, Number],
 | |
|       default: 'unset'
 | |
|     },
 | |
|     contentMarginTop: {
 | |
|       type: Number,
 | |
|       default: 50
 | |
|     },
 | |
|     zIndex: {
 | |
|       type: Number,
 | |
|       default: 60
 | |
|     },
 | |
|     bgOpacity: {
 | |
|       type: Number,
 | |
|       default: 75
 | |
|     }
 | |
|   },
 | |
|   data() {
 | |
|     return {
 | |
|       el: null,
 | |
|       content: null,
 | |
|       preventClickoutside: false,
 | |
|       isShowingPrompt: false
 | |
|     }
 | |
|   },
 | |
|   watch: {
 | |
|     show(newVal) {
 | |
|       if (newVal) {
 | |
|         this.$nextTick(this.setShow)
 | |
|       } else {
 | |
|         this.setHide()
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   computed: {
 | |
|     show: {
 | |
|       get() {
 | |
|         return this.value
 | |
|       },
 | |
|       set(val) {
 | |
|         this.$emit('input', val)
 | |
|       }
 | |
|     },
 | |
|     modalHeight() {
 | |
|       if (typeof this.height === 'string') {
 | |
|         return this.height
 | |
|       } else {
 | |
|         return this.height + 'px'
 | |
|       }
 | |
|     },
 | |
|     modalWidth() {
 | |
|       return typeof this.width === 'string' ? this.width : this.width + 'px'
 | |
|     }
 | |
|   },
 | |
|   methods: {
 | |
|     mousedownModal() {
 | |
|       this.preventClickoutside = true
 | |
|     },
 | |
|     mouseupModal() {
 | |
|       this.preventClickoutside = false
 | |
|     },
 | |
|     clickClose() {
 | |
|       this.show = false
 | |
|     },
 | |
|     clickBg(ev) {
 | |
|       if (!this.show || this.isShowingPrompt) return
 | |
|       if (this.preventClickoutside) {
 | |
|         this.preventClickoutside = false
 | |
|         return
 | |
|       }
 | |
|       if (this.processing && this.persistent) return
 | |
|       if (ev.srcElement && ev.srcElement.classList.contains('modal-bg')) {
 | |
|         this.show = false
 | |
|       }
 | |
|     },
 | |
|     hotkey(action) {
 | |
|       if (this.$store.state.innerModalOpen) return
 | |
|       if (action === this.$hotkeys.Modal.CLOSE) {
 | |
|         this.show = false
 | |
|       }
 | |
|     },
 | |
|     setShow() {
 | |
|       if (!this.el || !this.content) {
 | |
|         this.init()
 | |
|       }
 | |
|       if (!this.el || !this.content) {
 | |
|         return
 | |
|       }
 | |
| 
 | |
|       document.body.appendChild(this.el)
 | |
|       setTimeout(() => {
 | |
|         this.content.style.transform = 'scale(1)'
 | |
|       }, 10)
 | |
|       document.documentElement.classList.add('modal-open')
 | |
| 
 | |
|       this.$eventBus.$on('modal-hotkey', this.hotkey)
 | |
|       this.$store.commit('setOpenModal', this.name)
 | |
| 
 | |
|       // Set focus to the modal content
 | |
|       this.content.focus()
 | |
|     },
 | |
|     setHide() {
 | |
|       if (this.content) this.content.style.transform = 'scale(0)'
 | |
|       if (this.el) this.el.remove()
 | |
|       document.documentElement.classList.remove('modal-open')
 | |
| 
 | |
|       this.$eventBus.$off('modal-hotkey', this.hotkey)
 | |
|       this.$store.commit('setOpenModal', null)
 | |
|     },
 | |
|     init() {
 | |
|       this.el = this.$refs.wrapper
 | |
|       this.content = this.$refs.content
 | |
|       if (this.content && this.el) {
 | |
|         this.el.classList.remove('hidden')
 | |
|         this.el.classList.add('flex')
 | |
|         this.content.style.transform = 'scale(0)'
 | |
|         this.content.style.transition = 'transform 0.25s cubic-bezier(0.16, 1, 0.3, 1)'
 | |
|         this.el.style.opacity = 1
 | |
|         this.el.remove()
 | |
|       } else {
 | |
|         console.warn('Invalid modal init', this.name)
 | |
|       }
 | |
|     },
 | |
|     showingPrompt(isShowing) {
 | |
|       this.isShowingPrompt = isShowing
 | |
|     }
 | |
|   },
 | |
|   mounted() {
 | |
|     this.$eventBus.$on('showing-prompt', this.showingPrompt)
 | |
|   },
 | |
|   beforeDestroy() {
 | |
|     this.$eventBus.$off('showing-prompt', this.showingPrompt)
 | |
|   }
 | |
| }
 | |
| </script>
 |