mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-04 15:10:06 +00:00
204 lines
7.1 KiB
TypeScript
204 lines
7.1 KiB
TypeScript
import { defineStore } from 'pinia'
|
|
import { computed, ref } from 'vue'
|
|
|
|
import type { ComfyNodeDefImpl } from '@/stores/nodeDefStore'
|
|
import { useNodeDefStore } from '@/stores/nodeDefStore'
|
|
|
|
/** Helper class that defines how to construct a node from a model. */
|
|
export class ModelNodeProvider {
|
|
/** The node definition to use for this model. */
|
|
public nodeDef: ComfyNodeDefImpl
|
|
|
|
/** The node input key for where to insert the model name. */
|
|
public key: string
|
|
|
|
constructor(nodeDef: ComfyNodeDefImpl, key: string) {
|
|
this.nodeDef = nodeDef
|
|
this.key = key
|
|
}
|
|
}
|
|
|
|
/** Service for mapping model types (by folder name) to nodes. */
|
|
export const useModelToNodeStore = defineStore('modelToNode', () => {
|
|
const modelToNodeMap = ref<Record<string, ModelNodeProvider[]>>({})
|
|
const nodeDefStore = useNodeDefStore()
|
|
const haveDefaultsLoaded = ref(false)
|
|
|
|
/** Internal computed for reactive caching of registered node types */
|
|
const registeredNodeTypes = computed<Record<string, string>>(() => {
|
|
return Object.fromEntries(
|
|
Object.values(modelToNodeMap.value)
|
|
.flat()
|
|
.filter((provider) => !!provider.nodeDef)
|
|
.map((provider) => [provider.nodeDef.name, provider.key])
|
|
)
|
|
})
|
|
|
|
/** Internal computed for efficient reverse lookup: nodeType -> category */
|
|
const nodeTypeToCategory = computed(() => {
|
|
const lookup: Record<string, string> = {}
|
|
for (const [category, providers] of Object.entries(modelToNodeMap.value)) {
|
|
for (const provider of providers) {
|
|
// Extension nodes may not be installed
|
|
if (!provider.nodeDef) continue
|
|
// Only store the first category for each node type (matches current assetService behavior)
|
|
if (!lookup[provider.nodeDef.name]) {
|
|
lookup[provider.nodeDef.name] = category
|
|
}
|
|
}
|
|
}
|
|
return lookup
|
|
})
|
|
|
|
/** Get set of all registered node types for efficient lookup */
|
|
function getRegisteredNodeTypes(): Record<string, string> {
|
|
registerDefaults()
|
|
return registeredNodeTypes.value
|
|
}
|
|
|
|
/**
|
|
* Get the category for a given node type.
|
|
* Performs efficient O(1) lookup using cached reverse map.
|
|
* @param nodeType The node type name to find the category for
|
|
* @returns The category name, or undefined if not found
|
|
*/
|
|
function getCategoryForNodeType(nodeType: string): string | undefined {
|
|
registerDefaults()
|
|
|
|
// Handle invalid input gracefully
|
|
if (!nodeType || typeof nodeType !== 'string') {
|
|
return undefined
|
|
}
|
|
|
|
return nodeTypeToCategory.value[nodeType]
|
|
}
|
|
|
|
/**
|
|
* Find providers for modelType with hierarchical fallback.
|
|
* Tries exact match first, then falls back to top-level segment (e.g., "parent/child" → "parent").
|
|
* Note: Only falls back one level; "a/b/c" tries "a/b/c" then "a", not "a/b".
|
|
*/
|
|
function findProvidersWithFallback(
|
|
modelType: string
|
|
): ModelNodeProvider[] | undefined {
|
|
const exactMatch = modelToNodeMap.value[modelType]
|
|
if (exactMatch && exactMatch.length > 0) return exactMatch
|
|
|
|
const topLevel = modelType.split('/')[0]
|
|
if (topLevel === modelType) return undefined
|
|
|
|
const fallback = modelToNodeMap.value[topLevel]
|
|
|
|
if (fallback && fallback.length > 0) return fallback
|
|
|
|
return undefined
|
|
}
|
|
|
|
/**
|
|
* Get the node provider for the given model type name.
|
|
* Supports hierarchical lookups: if "parent/child" has no match, falls back to "parent".
|
|
* @param modelType The name of the model type to get the node provider for.
|
|
* @returns The node provider for the given model type name.
|
|
*/
|
|
function getNodeProvider(modelType: string): ModelNodeProvider | undefined {
|
|
registerDefaults()
|
|
return findProvidersWithFallback(modelType)?.[0]
|
|
}
|
|
|
|
/**
|
|
* Get the list of all valid node providers for the given model type name.
|
|
* Supports hierarchical lookups: if "parent/child" has no match, falls back to "parent".
|
|
* @param modelType The name of the model type to get the node providers for.
|
|
* @returns The list of all valid node providers for the given model type name.
|
|
*/
|
|
function getAllNodeProviders(modelType: string): ModelNodeProvider[] {
|
|
registerDefaults()
|
|
return findProvidersWithFallback(modelType) ?? []
|
|
}
|
|
/**
|
|
* Register a node provider for the given model type name.
|
|
* @param modelType The name of the model type to register the node provider for.
|
|
* @param nodeProvider The node provider to register.
|
|
*/
|
|
function registerNodeProvider(
|
|
modelType: string,
|
|
nodeProvider: ModelNodeProvider
|
|
) {
|
|
registerDefaults()
|
|
if (!nodeProvider.nodeDef) return
|
|
if (!modelToNodeMap.value[modelType]) {
|
|
modelToNodeMap.value[modelType] = []
|
|
}
|
|
modelToNodeMap.value[modelType].push(nodeProvider)
|
|
}
|
|
/**
|
|
* Register a node provider for the given simple names.
|
|
* @param modelType The name of the model type to register the node provider for.
|
|
* @param nodeClass The node class name to register.
|
|
* @param key The key to use for the node input.
|
|
*/
|
|
function quickRegister(modelType: string, nodeClass: string, key: string) {
|
|
registerNodeProvider(
|
|
modelType,
|
|
new ModelNodeProvider(nodeDefStore.nodeDefsByName[nodeClass], key)
|
|
)
|
|
}
|
|
|
|
function registerDefaults() {
|
|
if (haveDefaultsLoaded.value) {
|
|
return
|
|
}
|
|
if (Object.keys(nodeDefStore.nodeDefsByName).length === 0) {
|
|
return
|
|
}
|
|
haveDefaultsLoaded.value = true
|
|
|
|
quickRegister('checkpoints', 'CheckpointLoaderSimple', 'ckpt_name')
|
|
quickRegister('checkpoints', 'ImageOnlyCheckpointLoader', 'ckpt_name')
|
|
quickRegister('loras', 'LoraLoader', 'lora_name')
|
|
quickRegister('loras', 'LoraLoaderModelOnly', 'lora_name')
|
|
quickRegister('vae', 'VAELoader', 'vae_name')
|
|
quickRegister('controlnet', 'ControlNetLoader', 'control_net_name')
|
|
quickRegister('diffusion_models', 'UNETLoader', 'unet_name')
|
|
quickRegister('upscale_models', 'UpscaleModelLoader', 'model_name')
|
|
quickRegister('style_models', 'StyleModelLoader', 'style_model_name')
|
|
quickRegister('gligen', 'GLIGENLoader', 'gligen_name')
|
|
quickRegister('clip_vision', 'CLIPVisionLoader', 'clip_name')
|
|
quickRegister('text_encoders', 'CLIPLoader', 'clip_name')
|
|
quickRegister('audio_encoders', 'AudioEncoderLoader', 'audio_encoder_name')
|
|
quickRegister('model_patches', 'ModelPatchLoader', 'name')
|
|
quickRegister(
|
|
'animatediff_models',
|
|
'ADE_LoadAnimateDiffModel',
|
|
'model_name'
|
|
)
|
|
quickRegister(
|
|
'animatediff_motion_lora',
|
|
'ADE_AnimateDiffLoRALoader',
|
|
'name'
|
|
)
|
|
|
|
// Chatterbox TTS nodes: empty key means the node auto-loads models without
|
|
// a widget selector (createModelNodeFromAsset skips widget assignment)
|
|
quickRegister('chatterbox/chatterbox', 'FL_ChatterboxTTS', '')
|
|
quickRegister('chatterbox/chatterbox_turbo', 'FL_ChatterboxTurboTTS', '')
|
|
quickRegister(
|
|
'chatterbox/chatterbox_multilingual',
|
|
'FL_ChatterboxMultilingualTTS',
|
|
''
|
|
)
|
|
quickRegister('chatterbox/chatterbox_vc', 'FL_ChatterboxVC', '')
|
|
}
|
|
|
|
return {
|
|
modelToNodeMap,
|
|
getRegisteredNodeTypes,
|
|
getCategoryForNodeType,
|
|
getNodeProvider,
|
|
getAllNodeProviders,
|
|
registerNodeProvider,
|
|
quickRegister,
|
|
registerDefaults
|
|
}
|
|
})
|