Files
Memoh/apps/web/src/pages/chat/components/media-gallery-lightbox.vue
T
2026-03-07 15:22:51 +08:00

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>