mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
172 lines
4.7 KiB
Vue
172 lines
4.7 KiB
Vue
<template>
|
|
<Teleport to="body">
|
|
<Transition name="fade">
|
|
<div
|
|
v-if="isOpen"
|
|
ref="dialogRef"
|
|
class="fixed inset-0 z-[100] flex items-center justify-center bg-black/90"
|
|
role="dialog"
|
|
aria-modal="true"
|
|
:aria-label="currentItem?.name ? `Media preview: ${currentItem.name}` : 'Media preview'"
|
|
tabindex="-1"
|
|
@click.self="close"
|
|
>
|
|
<!-- Close -->
|
|
<button
|
|
ref="closeButtonRef"
|
|
type="button"
|
|
class="absolute right-4 top-4 z-10 rounded-full p-2 text-white/80 hover:bg-white/10 hover:text-white transition-colors"
|
|
aria-label="Close"
|
|
@click="close"
|
|
>
|
|
<FontAwesomeIcon
|
|
:icon="['fas', 'xmark']"
|
|
class="size-6"
|
|
/>
|
|
</button>
|
|
|
|
<!-- Prev -->
|
|
<button
|
|
v-if="items.length > 1"
|
|
type="button"
|
|
class="absolute left-4 top-1/2 -translate-y-1/2 z-10 rounded-full p-3 text-white/80 hover:bg-white/10 hover:text-white transition-colors"
|
|
aria-label="Previous"
|
|
@click.stop="prev"
|
|
>
|
|
<FontAwesomeIcon
|
|
:icon="['fas', 'chevron-left']"
|
|
class="size-6"
|
|
/>
|
|
</button>
|
|
|
|
<!-- Next -->
|
|
<button
|
|
v-if="items.length > 1"
|
|
type="button"
|
|
class="absolute right-4 top-1/2 -translate-y-1/2 z-10 rounded-full p-3 text-white/80 hover:bg-white/10 hover:text-white transition-colors"
|
|
aria-label="Next"
|
|
@click.stop="next"
|
|
>
|
|
<FontAwesomeIcon
|
|
:icon="['fas', 'chevron-right']"
|
|
class="size-6"
|
|
/>
|
|
</button>
|
|
|
|
<!-- Media content -->
|
|
<div class="max-w-[90vw] max-h-[90vh] flex items-center justify-center">
|
|
<img
|
|
v-if="currentItem?.type === 'image'"
|
|
:src="currentItem.src"
|
|
:alt="currentItem.name || 'image'"
|
|
class="max-w-full max-h-[90vh] object-contain select-none"
|
|
draggable="false"
|
|
@click.stop
|
|
>
|
|
<video
|
|
v-else-if="currentItem?.type === 'video'"
|
|
:src="currentItem.src"
|
|
controls
|
|
class="max-w-full max-h-[90vh] object-contain"
|
|
@click.stop
|
|
/>
|
|
</div>
|
|
|
|
<!-- Counter -->
|
|
<div
|
|
v-if="items.length > 1"
|
|
class="absolute bottom-4 left-1/2 -translate-x-1/2 px-3 py-1 rounded-full bg-black/50 text-white/90 text-sm"
|
|
>
|
|
{{ (openIndex ?? 0) + 1 }} / {{ items.length }}
|
|
</div>
|
|
</div>
|
|
</Transition>
|
|
</Teleport>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed, watchEffect, onUnmounted, nextTick, ref } from 'vue'
|
|
|
|
export interface MediaGalleryItem {
|
|
src: string
|
|
type: 'image' | 'video'
|
|
name?: string
|
|
}
|
|
|
|
const props = defineProps<{
|
|
items: MediaGalleryItem[]
|
|
openIndex: number | null
|
|
}>()
|
|
|
|
const emit = defineEmits<{
|
|
'update:openIndex': [value: number | null]
|
|
}>()
|
|
|
|
const isOpen = computed(() => props.openIndex !== null && props.items.length > 0)
|
|
const closeButtonRef = ref<HTMLButtonElement | null>(null)
|
|
const dialogRef = ref<HTMLElement | null>(null)
|
|
let previousFocusedElement: HTMLElement | null = null
|
|
|
|
const currentItem = computed(() => {
|
|
const idx = props.openIndex
|
|
if (idx === null || idx < 0 || idx >= props.items.length) return null
|
|
return props.items[idx] ?? null
|
|
})
|
|
|
|
function close() {
|
|
emit('update:openIndex', null)
|
|
}
|
|
|
|
function prev() {
|
|
if (props.openIndex === null) return
|
|
const idx = props.openIndex <= 0 ? props.items.length - 1 : props.openIndex - 1
|
|
emit('update:openIndex', idx)
|
|
}
|
|
|
|
function next() {
|
|
if (props.openIndex === null) return
|
|
const idx = props.openIndex >= props.items.length - 1 ? 0 : props.openIndex + 1
|
|
emit('update:openIndex', idx)
|
|
}
|
|
|
|
function handleKeydown(e: KeyboardEvent) {
|
|
if (props.openIndex === null) return
|
|
if (e.key === 'Escape') close()
|
|
else if (e.key === 'ArrowLeft') prev()
|
|
else if (e.key === 'ArrowRight') next()
|
|
}
|
|
|
|
let removeListener: (() => void) | null = null
|
|
|
|
watchEffect(() => {
|
|
if (isOpen.value) {
|
|
previousFocusedElement = document.activeElement instanceof HTMLElement ? document.activeElement : null
|
|
window.addEventListener('keydown', handleKeydown)
|
|
removeListener = () => window.removeEventListener('keydown', handleKeydown)
|
|
nextTick(() => {
|
|
closeButtonRef.value?.focus()
|
|
})
|
|
} else if (removeListener) {
|
|
removeListener()
|
|
removeListener = null
|
|
previousFocusedElement?.focus()
|
|
previousFocusedElement = null
|
|
}
|
|
})
|
|
|
|
onUnmounted(() => {
|
|
removeListener?.()
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition: opacity 0.2s ease;
|
|
}
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
}
|
|
</style>
|