mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-28 18:54:09 +00:00
## Summary Allows SaveImage and other nodes with ImagePreviews to be dragged by the image. ┆Issue is synchronized with this [Notion page](https://www.notion.so/PR-6388-Fix-Disable-pointer-events-on-images-to-allow-them-to-be-drag-targets-29c6d73d365081e796f0e8171a860062) by [Unito](https://www.unito.io)
288 lines
8.1 KiB
Vue
288 lines
8.1 KiB
Vue
<template>
|
|
<div
|
|
v-if="imageUrls.length > 0"
|
|
class="image-preview group relative flex size-full min-h-16 min-w-16 flex-col px-2"
|
|
tabindex="0"
|
|
role="region"
|
|
:aria-label="$t('g.imagePreview')"
|
|
@mouseenter="handleMouseEnter"
|
|
@mouseleave="handleMouseLeave"
|
|
@keydown="handleKeyDown"
|
|
>
|
|
<!-- Image Wrapper -->
|
|
<div
|
|
class="relative h-88 w-full grow overflow-hidden rounded-[5px] bg-node-component-surface"
|
|
>
|
|
<!-- Error State -->
|
|
<div
|
|
v-if="imageError"
|
|
class="flex size-full flex-col items-center justify-center bg-smoke-800/50 text-center text-white"
|
|
>
|
|
<i class="mb-2 icon-[lucide--image-off] h-12 w-12 text-smoke-400" />
|
|
<p class="text-sm text-smoke-300">{{ $t('g.imageFailedToLoad') }}</p>
|
|
<p class="mt-1 text-xs text-smoke-400">
|
|
{{ getImageFilename(currentImageUrl) }}
|
|
</p>
|
|
</div>
|
|
|
|
<!-- Loading State -->
|
|
<Skeleton v-else-if="isLoading" class="size-full" border-radius="5px" />
|
|
|
|
<!-- Main Image -->
|
|
<img
|
|
v-else
|
|
ref="currentImageEl"
|
|
:src="currentImageUrl"
|
|
:alt="imageAltText"
|
|
class="block size-full object-contain pointer-events-none"
|
|
@load="handleImageLoad"
|
|
@error="handleImageError"
|
|
/>
|
|
|
|
<!-- Floating Action Buttons (appear on hover) -->
|
|
<div v-if="isHovered" class="actions absolute top-2 right-2 flex gap-2.5">
|
|
<!-- Mask/Edit Button -->
|
|
<button
|
|
v-if="!hasMultipleImages"
|
|
:class="actionButtonClass"
|
|
:title="$t('g.editOrMaskImage')"
|
|
:aria-label="$t('g.editOrMaskImage')"
|
|
@click="handleEditMask"
|
|
>
|
|
<i-comfy:mask class="h-4 w-4" />
|
|
</button>
|
|
|
|
<!-- Download Button -->
|
|
<button
|
|
:class="actionButtonClass"
|
|
:title="$t('g.downloadImage')"
|
|
:aria-label="$t('g.downloadImage')"
|
|
@click="handleDownload"
|
|
>
|
|
<i class="icon-[lucide--download] h-4 w-4" />
|
|
</button>
|
|
|
|
<!-- Close Button -->
|
|
<button
|
|
:class="actionButtonClass"
|
|
:title="$t('g.removeImage')"
|
|
:aria-label="$t('g.removeImage')"
|
|
@click="handleRemove"
|
|
>
|
|
<i class="icon-[lucide--x] h-4 w-4" />
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Multiple Images Navigation -->
|
|
<div
|
|
v-if="hasMultipleImages"
|
|
class="absolute right-2 bottom-2 left-2 flex justify-center gap-1"
|
|
>
|
|
<button
|
|
v-for="(_, index) in imageUrls"
|
|
:key="index"
|
|
:class="getNavigationDotClass(index)"
|
|
:aria-label="
|
|
$t('g.viewImageOfTotal', {
|
|
index: index + 1,
|
|
total: imageUrls.length
|
|
})
|
|
"
|
|
@click="setCurrentIndex(index)"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="relative">
|
|
<!-- Image Dimensions -->
|
|
<div class="mt-2 text-center text-xs text-white">
|
|
<span v-if="imageError" class="text-red-400">
|
|
{{ $t('g.errorLoadingImage') }}
|
|
</span>
|
|
<span v-else-if="isLoading" class="text-smoke-400">
|
|
{{ $t('g.loading') }}...
|
|
</span>
|
|
<span v-else>
|
|
{{ actualDimensions || $t('g.calculatingDimensions') }}
|
|
</span>
|
|
</div>
|
|
<LODFallback />
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { useToast } from 'primevue'
|
|
import Skeleton from 'primevue/skeleton'
|
|
import { computed, ref, watch } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
|
|
import { downloadFile } from '@/base/common/downloadUtil'
|
|
import { app } from '@/scripts/app'
|
|
import { useCommandStore } from '@/stores/commandStore'
|
|
import { useNodeOutputStore } from '@/stores/imagePreviewStore'
|
|
|
|
import LODFallback from './LODFallback.vue'
|
|
|
|
interface ImagePreviewProps {
|
|
/** Array of image URLs to display */
|
|
readonly imageUrls: readonly string[]
|
|
/** Optional node ID for context-aware actions */
|
|
readonly nodeId?: string
|
|
}
|
|
|
|
const props = defineProps<ImagePreviewProps>()
|
|
|
|
const { t } = useI18n()
|
|
const commandStore = useCommandStore()
|
|
const nodeOutputStore = useNodeOutputStore()
|
|
|
|
const actionButtonClass =
|
|
'flex h-8 min-h-8 items-center justify-center gap-2.5 rounded-lg border-0 bg-button-surface px-2 py-2 text-button-surface-contrast shadow-sm transition-colors duration-200 hover:bg-button-hover-surface focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-button-surface-contrast focus-visible:ring-offset-2 focus-visible:ring-offset-transparent cursor-pointer'
|
|
|
|
// Component state
|
|
const currentIndex = ref(0)
|
|
const isHovered = ref(false)
|
|
const actualDimensions = ref<string | null>(null)
|
|
const imageError = ref(false)
|
|
const isLoading = ref(false)
|
|
|
|
const currentImageEl = ref<HTMLImageElement>()
|
|
|
|
// Computed values
|
|
const currentImageUrl = computed(() => props.imageUrls[currentIndex.value])
|
|
const hasMultipleImages = computed(() => props.imageUrls.length > 1)
|
|
const imageAltText = computed(() => `Node output ${currentIndex.value + 1}`)
|
|
|
|
// Watch for URL changes and reset state
|
|
watch(
|
|
() => props.imageUrls,
|
|
(newUrls) => {
|
|
// Reset current index if it's out of bounds
|
|
if (currentIndex.value >= newUrls.length) {
|
|
currentIndex.value = 0
|
|
}
|
|
|
|
// Reset loading and error states when URLs change
|
|
actualDimensions.value = null
|
|
imageError.value = false
|
|
isLoading.value = false
|
|
},
|
|
{ deep: true }
|
|
)
|
|
|
|
// Event handlers
|
|
const handleImageLoad = (event: Event) => {
|
|
if (!event.target || !(event.target instanceof HTMLImageElement)) return
|
|
const img = event.target
|
|
isLoading.value = false
|
|
imageError.value = false
|
|
if (img.naturalWidth && img.naturalHeight) {
|
|
actualDimensions.value = `${img.naturalWidth} x ${img.naturalHeight}`
|
|
}
|
|
}
|
|
|
|
const handleImageError = () => {
|
|
isLoading.value = false
|
|
imageError.value = true
|
|
actualDimensions.value = null
|
|
}
|
|
|
|
// In vueNodes mode, we need to set them manually before opening the mask editor.
|
|
const setupNodeForMaskEditor = () => {
|
|
if (!props.nodeId || !currentImageEl.value) return
|
|
const node = app.rootGraph?.getNodeById(props.nodeId)
|
|
if (!node) return
|
|
node.imageIndex = currentIndex.value
|
|
node.imgs = [currentImageEl.value]
|
|
app.canvas?.select(node)
|
|
}
|
|
|
|
const handleEditMask = () => {
|
|
setupNodeForMaskEditor()
|
|
void commandStore.execute('Comfy.MaskEditor.OpenMaskEditor')
|
|
}
|
|
|
|
const handleDownload = () => {
|
|
try {
|
|
downloadFile(currentImageUrl.value)
|
|
} catch (error) {
|
|
useToast().add({
|
|
severity: 'error',
|
|
summary: 'Error',
|
|
detail: t('g.failedToDownloadImage'),
|
|
life: 3000,
|
|
group: 'image-preview'
|
|
})
|
|
}
|
|
}
|
|
|
|
const handleRemove = () => {
|
|
if (!props.nodeId) return
|
|
nodeOutputStore.removeNodeOutputs(props.nodeId)
|
|
}
|
|
|
|
const setCurrentIndex = (index: number) => {
|
|
if (index >= 0 && index < props.imageUrls.length) {
|
|
currentIndex.value = index
|
|
actualDimensions.value = null
|
|
isLoading.value = true
|
|
imageError.value = false
|
|
}
|
|
}
|
|
|
|
const handleMouseEnter = () => {
|
|
isHovered.value = true
|
|
}
|
|
|
|
const handleMouseLeave = () => {
|
|
isHovered.value = false
|
|
}
|
|
|
|
const getNavigationDotClass = (index: number) => {
|
|
return [
|
|
'w-2 h-2 rounded-full transition-all duration-200 border-0 cursor-pointer',
|
|
index === currentIndex.value ? 'bg-white' : 'bg-white/50 hover:bg-white/80'
|
|
]
|
|
}
|
|
|
|
const handleKeyDown = (event: KeyboardEvent) => {
|
|
if (props.imageUrls.length <= 1) return
|
|
|
|
switch (event.key) {
|
|
case 'ArrowLeft':
|
|
event.preventDefault()
|
|
setCurrentIndex(
|
|
currentIndex.value > 0
|
|
? currentIndex.value - 1
|
|
: props.imageUrls.length - 1
|
|
)
|
|
break
|
|
case 'ArrowRight':
|
|
event.preventDefault()
|
|
setCurrentIndex(
|
|
currentIndex.value < props.imageUrls.length - 1
|
|
? currentIndex.value + 1
|
|
: 0
|
|
)
|
|
break
|
|
case 'Home':
|
|
event.preventDefault()
|
|
setCurrentIndex(0)
|
|
break
|
|
case 'End':
|
|
event.preventDefault()
|
|
setCurrentIndex(props.imageUrls.length - 1)
|
|
break
|
|
}
|
|
}
|
|
|
|
const getImageFilename = (url: string): string => {
|
|
try {
|
|
return new URL(url).searchParams.get('filename') || 'Unknown file'
|
|
} catch {
|
|
return 'Invalid URL'
|
|
}
|
|
}
|
|
</script>
|