mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-09 17:40:09 +00:00
* basic/empty model library sidebar tab in-progress * make it actually list out models * extremely primitive search impl * list out available folders (incomplete list atm) * load list dynamically * nice lil loading icon * that's not doing anything * run autoformatter * fix up some absolute vue shenanigans * swap to pi-box * is_fake_object * i think apply the tailwind thingo * trim '.safetensors' from end of display title * oop * after load, retain title if no new title is given * is_load_requested to prevent duplication * dirty initial model metadata load & preview based on node preview code * update model store tests * initial image icon for model lib * i hate this * better empty spacer * add api handler for '/models' * load model folders list instead of hardcoding * add a 'no content' placeholder for empty folders * autoformat * autoload model metadata * error handling on metadata loading * larger model icons * click a model to spawn a node for it * draggable model nodes * add a setting for whether to autoload or not * autoformat will be the death of me * cleanup promise code * make the model preview actually half-decent * revert bad unchecked change * put registration back
169 lines
5.2 KiB
TypeScript
169 lines
5.2 KiB
TypeScript
import { api } from '@/scripts/api'
|
|
import { defineStore } from 'pinia'
|
|
|
|
/** (Internal helper) finds a value in a metadata object from any of a list of keys. */
|
|
function _findInMetadata(metadata: any, ...keys: string[]): string | null {
|
|
for (const key of keys) {
|
|
if (key in metadata) {
|
|
return metadata[key]
|
|
}
|
|
for (const k in metadata) {
|
|
if (k.endsWith(key)) {
|
|
return metadata[k]
|
|
}
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
/** Defines and holds metadata for a model */
|
|
export class ComfyModelDef {
|
|
/** Proper filename of the model */
|
|
name: string = ''
|
|
/** Directory containing the model, eg 'checkpoints' */
|
|
directory: string = ''
|
|
/** Title / display name of the model, sometimes same as the name but not always */
|
|
title: string = ''
|
|
/** Metadata: architecture ID for the model, such as 'stable-diffusion-xl-v1-base' */
|
|
architecture_id: string = ''
|
|
/** Metadata: author of the model */
|
|
author: string = ''
|
|
/** Metadata: resolution of the model, eg '1024x1024' */
|
|
resolution: string = ''
|
|
/** Metadata: description of the model */
|
|
description: string = ''
|
|
/** Metadata: usage hint for the model */
|
|
usage_hint: string = ''
|
|
/** Metadata: trigger phrase for the model */
|
|
trigger_phrase: string = ''
|
|
/** Metadata: tags list for the model */
|
|
tags: string[] = []
|
|
/** Metadata: image for the model */
|
|
image: string = ''
|
|
/** Whether the model metadata has been loaded from the server, used for `load()` */
|
|
has_loaded_metadata: boolean = false
|
|
/** If true, a metadata load request has been triggered, but may or may not yet have finished loading */
|
|
is_load_requested: boolean = false
|
|
/** If true, this is a fake model object used as a placeholder for something (eg a loading icon) */
|
|
is_fake_object: boolean = false
|
|
|
|
constructor(name: string, directory: string) {
|
|
this.name = name
|
|
this.title = name.replaceAll('\\', '/').split('/').pop()
|
|
if (this.title.endsWith('.safetensors')) {
|
|
this.title = this.title.slice(0, -'.safetensors'.length)
|
|
}
|
|
this.directory = directory
|
|
}
|
|
|
|
/** Loads the model metadata from the server, filling in this object if data is available */
|
|
async load(): Promise<void> {
|
|
if (this.has_loaded_metadata || this.is_load_requested) {
|
|
return
|
|
}
|
|
this.is_load_requested = true
|
|
try {
|
|
const metadata = await api.viewMetadata(this.directory, this.name)
|
|
if (!metadata) {
|
|
return
|
|
}
|
|
this.title =
|
|
_findInMetadata(
|
|
metadata,
|
|
'modelspec.title',
|
|
'title',
|
|
'display_name',
|
|
'name'
|
|
) || this.title
|
|
this.architecture_id =
|
|
_findInMetadata(metadata, 'modelspec.architecture', 'architecture') ||
|
|
''
|
|
this.author =
|
|
_findInMetadata(metadata, 'modelspec.author', 'author') || ''
|
|
this.description =
|
|
_findInMetadata(metadata, 'modelspec.description', 'description') || ''
|
|
this.resolution =
|
|
_findInMetadata(metadata, 'modelspec.resolution', 'resolution') || ''
|
|
this.usage_hint =
|
|
_findInMetadata(metadata, 'modelspec.usage_hint', 'usage_hint') || ''
|
|
this.trigger_phrase =
|
|
_findInMetadata(
|
|
metadata,
|
|
'modelspec.trigger_phrase',
|
|
'trigger_phrase'
|
|
) || ''
|
|
this.image =
|
|
_findInMetadata(
|
|
metadata,
|
|
'modelspec.thumbnail',
|
|
'thumbnail',
|
|
'image',
|
|
'icon'
|
|
) || ''
|
|
const tagsCommaSeparated =
|
|
_findInMetadata(metadata, 'modelspec.tags', 'tags') || ''
|
|
this.tags = tagsCommaSeparated.split(',').map((tag) => tag.trim())
|
|
this.has_loaded_metadata = true
|
|
} catch (error) {
|
|
console.error('Error loading model metadata', this.name, this, error)
|
|
}
|
|
}
|
|
}
|
|
|
|
/** Model store for a folder */
|
|
export class ModelStore {
|
|
models: Record<string, ComfyModelDef> = {}
|
|
|
|
constructor(directory: string, models: string[]) {
|
|
for (const model of models) {
|
|
this.models[model] = new ComfyModelDef(model, directory)
|
|
}
|
|
}
|
|
|
|
async loadModelMetadata(modelName: string) {
|
|
if (this.models[modelName]) {
|
|
await this.models[modelName].load()
|
|
}
|
|
}
|
|
}
|
|
|
|
const folderBlacklist = ['configs', 'custom_nodes']
|
|
|
|
/** Model store handler, wraps individual per-folder model stores */
|
|
export const useModelStore = defineStore('modelStore', {
|
|
state: () => ({
|
|
modelStoreMap: {} as Record<string, ModelStore>,
|
|
isLoading: {} as Record<string, Promise<ModelStore>>,
|
|
modelFolders: [] as string[]
|
|
}),
|
|
actions: {
|
|
async getModelsInFolderCached(folder: string): Promise<ModelStore> {
|
|
if (folder in this.modelStoreMap) {
|
|
return this.modelStoreMap[folder]
|
|
}
|
|
if (this.isLoading[folder]) {
|
|
return this.isLoading[folder]
|
|
}
|
|
const promise = api.getModels(folder).then((models) => {
|
|
if (!models) {
|
|
return null
|
|
}
|
|
const store = new ModelStore(folder, models)
|
|
this.modelStoreMap[folder] = store
|
|
this.isLoading[folder] = false
|
|
return store
|
|
})
|
|
this.isLoading[folder] = promise
|
|
return promise
|
|
},
|
|
clearCache() {
|
|
this.modelStoreMap = {}
|
|
},
|
|
async getModelFolders() {
|
|
this.modelFolders = (await api.getModelFolders()).filter(
|
|
(folder) => !folderBlacklist.includes(folder)
|
|
)
|
|
}
|
|
}
|
|
})
|