Files
ComfyUI_frontend/src/composables/node/useNodeMediaUpload.ts
2025-06-09 03:52:17 -07:00

123 lines
3.3 KiB
TypeScript

import type { LGraphNode } from '@comfyorg/litegraph'
import { useMediaLoaderWidget } from '@/composables/widgets/useMediaLoaderWidget'
import type { InputSpec } from '@/schemas/nodeDef/nodeDefSchemaV2'
import { api } from '@/scripts/api'
import { useToastStore } from '@/stores/toastStore'
const MEDIA_LOADER_WIDGET_NAME = '$$node-media-loader'
const PASTED_IMAGE_EXPIRY_MS = 2000
const uploadFile = async (file: File, isPasted: boolean) => {
const body = new FormData()
body.append('image', file)
if (isPasted) body.append('subfolder', 'pasted')
const resp = await api.fetchApi('/upload/image', {
method: 'POST',
body
})
if (resp.status !== 200) {
useToastStore().addAlert(resp.status + ' - ' + resp.statusText)
return
}
const data = await resp.json()
return data.subfolder ? `${data.subfolder}/${data.name}` : data.name
}
interface MediaUploadOptions {
fileFilter?: (file: File) => boolean
onUploadComplete: (paths: string[]) => void
allow_batch?: boolean
accept?: string
}
/**
* Composable for handling media upload with Vue MediaLoader widget
*/
export function useNodeMediaUpload() {
const mediaLoaderWidget = useMediaLoaderWidget()
const findMediaLoaderWidget = (node: LGraphNode) =>
node.widgets?.find((w) => w.name === MEDIA_LOADER_WIDGET_NAME)
const addMediaLoaderWidget = (
node: LGraphNode,
options: MediaUploadOptions
) => {
const isPastedFile = (file: File): boolean =>
file.name === 'image.png' &&
file.lastModified - Date.now() < PASTED_IMAGE_EXPIRY_MS
const handleUpload = async (file: File) => {
try {
const path = await uploadFile(file, isPastedFile(file))
if (!path) return
return path
} catch (error) {
useToastStore().addAlert(String(error))
}
}
// Create the MediaLoader widget
const widget = mediaLoaderWidget(node, {
name: MEDIA_LOADER_WIDGET_NAME,
type: 'MEDIA_LOADER'
} as InputSpec)
// Connect the widget to the upload handler
if (widget.options) {
;(widget.options as any).onFilesSelected = async (files: File[]) => {
const filteredFiles = options.fileFilter
? files.filter(options.fileFilter)
: files
const paths = await Promise.all(filteredFiles.map(handleUpload))
const validPaths = paths.filter((p): p is string => !!p)
if (validPaths.length) {
options.onUploadComplete(validPaths)
}
}
}
return widget
}
/**
* Shows media loader widget for a node
* @param node The graph node to show the widget for
* @param options Upload configuration options
*/
function showMediaLoader(node: LGraphNode, options: MediaUploadOptions) {
const widget =
findMediaLoaderWidget(node) ?? addMediaLoaderWidget(node, options)
node.setDirtyCanvas?.(true)
return widget
}
/**
* Removes media loader widget from a node
* @param node The graph node to remove the widget from
*/
function removeMediaLoader(node: LGraphNode) {
if (!node.widgets) return
const widgetIdx = node.widgets.findIndex(
(w) => w.name === MEDIA_LOADER_WIDGET_NAME
)
if (widgetIdx > -1) {
node.widgets[widgetIdx].onRemove?.()
node.widgets.splice(widgetIdx, 1)
}
}
return {
showMediaLoader,
removeMediaLoader,
addMediaLoaderWidget
}
}