mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-01-26 19:09:52 +00:00
* fix: vite primevue/treenode import error * refactor: remove useless @ts-ignore and replace with @ts-expect-error * build(tsconfig): enable incremental to speed up secondary time type check
485 lines
11 KiB
TypeScript
485 lines
11 KiB
TypeScript
import type { LiteGraph, LGraphCanvas } from '@comfyorg/litegraph'
|
|
|
|
/**
|
|
* @typedef { import("./src/scripts/app")["app"] } app
|
|
* @typedef { import("../../src/types/litegraph") } LG
|
|
* @typedef { import("../../src/types/litegraph").IWidget } IWidget
|
|
* @typedef { import("../../src/types/litegraph").ContextMenuItem } ContextMenuItem
|
|
* @typedef { import("../../src/types/litegraph").INodeInputSlot } INodeInputSlot
|
|
* @typedef { import("../../src/types/litegraph").INodeOutputSlot } INodeOutputSlot
|
|
* @typedef { InstanceType<LG["LGraphNode"]> & { widgets?: Array<IWidget> } } LGNode
|
|
* @typedef { (...args: EzOutput[] | [...EzOutput[], Record<string, unknown>]) => EzNode } EzNodeFactory
|
|
*/
|
|
|
|
export type EzNameSpace = Record<string, (...args) => EzNode>
|
|
|
|
export class EzConnection {
|
|
/** @type { app } */
|
|
app
|
|
/** @type { InstanceType<LG["LLink"]> } */
|
|
link
|
|
|
|
get originNode() {
|
|
return new EzNode(this.app, this.app.graph.getNodeById(this.link.origin_id))
|
|
}
|
|
|
|
get originOutput() {
|
|
return this.originNode.outputs[this.link.origin_slot]
|
|
}
|
|
|
|
get targetNode() {
|
|
return new EzNode(this.app, this.app.graph.getNodeById(this.link.target_id))
|
|
}
|
|
|
|
get targetInput() {
|
|
return this.targetNode.inputs[this.link.target_slot]
|
|
}
|
|
|
|
/**
|
|
* @param { app } app
|
|
* @param { InstanceType<LG["LLink"]> } link
|
|
*/
|
|
constructor(app, link) {
|
|
this.app = app
|
|
this.link = link
|
|
}
|
|
|
|
disconnect() {
|
|
this.targetInput.disconnect()
|
|
}
|
|
}
|
|
|
|
export class EzSlot {
|
|
/** @type { EzNode } */
|
|
node
|
|
/** @type { number } */
|
|
index
|
|
|
|
/**
|
|
* @param { EzNode } node
|
|
* @param { number } index
|
|
*/
|
|
constructor(node, index) {
|
|
this.node = node
|
|
this.index = index
|
|
}
|
|
}
|
|
|
|
export class EzInput extends EzSlot {
|
|
/** @type { INodeInputSlot } */
|
|
input
|
|
|
|
/**
|
|
* @param { EzNode } node
|
|
* @param { number } index
|
|
* @param { INodeInputSlot } input
|
|
*/
|
|
constructor(node, index, input) {
|
|
super(node, index)
|
|
this.input = input
|
|
}
|
|
|
|
get connection() {
|
|
const link = this.node.node.inputs?.[this.index]?.link
|
|
if (link == null) {
|
|
return null
|
|
}
|
|
return new EzConnection(this.node.app, this.node.app.graph.links[link])
|
|
}
|
|
|
|
disconnect() {
|
|
this.node.node.disconnectInput(this.index)
|
|
}
|
|
}
|
|
|
|
export class EzOutput extends EzSlot {
|
|
/** @type { INodeOutputSlot } */
|
|
output
|
|
|
|
/**
|
|
* @param { EzNode } node
|
|
* @param { number } index
|
|
* @param { INodeOutputSlot } output
|
|
*/
|
|
constructor(node, index, output) {
|
|
super(node, index)
|
|
this.output = output
|
|
}
|
|
|
|
get connections() {
|
|
return (this.node.node.outputs?.[this.index]?.links ?? []).map(
|
|
(l) => new EzConnection(this.node.app, this.node.app.graph.links[l])
|
|
)
|
|
}
|
|
|
|
/**
|
|
* @param { EzInput } input
|
|
*/
|
|
connectTo(input) {
|
|
if (!input) throw new Error('Invalid input')
|
|
|
|
/**
|
|
* @type { LG["LLink"] | null }
|
|
*/
|
|
const link = this.node.node.connect(
|
|
this.index,
|
|
input.node.node,
|
|
input.index
|
|
)
|
|
if (!link) {
|
|
const inp = input.input
|
|
const inName = inp.name || inp.label || inp.type
|
|
throw new Error(
|
|
`Connecting from ${input.node.node.type}#${input.node.id}[${inName}#${input.index}] -> ${this.node.node.type}#${this.node.id}[${
|
|
this.output.name ?? this.output.type
|
|
}#${this.index}] failed.`
|
|
)
|
|
}
|
|
return link
|
|
}
|
|
}
|
|
|
|
export class EzNodeMenuItem {
|
|
/** @type { EzNode } */
|
|
node
|
|
/** @type { number } */
|
|
index
|
|
/** @type { ContextMenuItem } */
|
|
item
|
|
|
|
/**
|
|
* @param { EzNode } node
|
|
* @param { number } index
|
|
* @param { ContextMenuItem } item
|
|
*/
|
|
constructor(node, index, item) {
|
|
this.node = node
|
|
this.index = index
|
|
this.item = item
|
|
}
|
|
|
|
call(selectNode = true) {
|
|
if (!this.item?.callback)
|
|
throw new Error(
|
|
`Menu Item ${this.item?.content ?? '[null]'} has no callback.`
|
|
)
|
|
if (selectNode) {
|
|
this.node.select()
|
|
}
|
|
return this.item.callback.call(
|
|
this.node.node,
|
|
undefined,
|
|
undefined,
|
|
undefined,
|
|
undefined,
|
|
this.node.node
|
|
)
|
|
}
|
|
}
|
|
|
|
export class EzWidget {
|
|
/** @type { EzNode } */
|
|
node
|
|
/** @type { number } */
|
|
index
|
|
/** @type { IWidget } */
|
|
widget
|
|
|
|
/**
|
|
* @param { EzNode } node
|
|
* @param { number } index
|
|
* @param { IWidget } widget
|
|
*/
|
|
constructor(node, index, widget) {
|
|
this.node = node
|
|
this.index = index
|
|
this.widget = widget
|
|
}
|
|
|
|
get value() {
|
|
return this.widget.value
|
|
}
|
|
|
|
set value(v) {
|
|
this.widget.value = v
|
|
this.widget.callback?.call?.(this.widget, v)
|
|
}
|
|
|
|
get isConvertedToInput() {
|
|
return this.widget.type === 'converted-widget'
|
|
}
|
|
|
|
getConvertedInput() {
|
|
if (!this.isConvertedToInput)
|
|
throw new Error(`Widget ${this.widget.name} is not converted to input.`)
|
|
|
|
return this.node.inputs.find(
|
|
(inp) => inp.input['widget']?.name === this.widget.name
|
|
)
|
|
}
|
|
|
|
convertToWidget() {
|
|
if (!this.isConvertedToInput)
|
|
throw new Error(
|
|
`Widget ${this.widget.name} cannot be converted as it is already a widget.`
|
|
)
|
|
var menu = this.node.menu['Convert Input to Widget'].item.submenu.options
|
|
var index = menu.findIndex(
|
|
(a) => a.content == `Convert ${this.widget.name} to widget`
|
|
)
|
|
menu[index].callback.call()
|
|
}
|
|
|
|
convertToInput() {
|
|
if (this.isConvertedToInput)
|
|
throw new Error(
|
|
`Widget ${this.widget.name} cannot be converted as it is already an input.`
|
|
)
|
|
var menu = this.node.menu['Convert Widget to Input'].item.submenu.options
|
|
var index = menu.findIndex(
|
|
(a) => a.content == `Convert ${this.widget.name} to input`
|
|
)
|
|
menu[index].callback.call()
|
|
}
|
|
}
|
|
|
|
export class EzNode {
|
|
/** @type { app } */
|
|
app
|
|
/** @type { LGNode } */
|
|
node
|
|
|
|
/**
|
|
* @param { app } app
|
|
* @param { LGNode } node
|
|
*/
|
|
constructor(app, node) {
|
|
this.app = app
|
|
this.node = node
|
|
}
|
|
|
|
get id() {
|
|
return this.node.id
|
|
}
|
|
|
|
get inputs() {
|
|
return this.#makeLookupArray('inputs', 'name', EzInput)
|
|
}
|
|
|
|
get outputs() {
|
|
return this.#makeLookupArray('outputs', 'name', EzOutput)
|
|
}
|
|
|
|
get widgets() {
|
|
return this.#makeLookupArray('widgets', 'name', EzWidget)
|
|
}
|
|
|
|
get menu() {
|
|
return this.#makeLookupArray(
|
|
() => this.app.canvas.getNodeMenuOptions(this.node),
|
|
'content',
|
|
EzNodeMenuItem
|
|
)
|
|
}
|
|
|
|
get isRemoved() {
|
|
return !this.app.graph.getNodeById(this.id)
|
|
}
|
|
|
|
select(addToSelection = false) {
|
|
this.app.canvas.selectNode(this.node, addToSelection)
|
|
}
|
|
|
|
// /**
|
|
// * @template { "inputs" | "outputs" } T
|
|
// * @param { T } type
|
|
// * @returns { Record<string, type extends "inputs" ? EzInput : EzOutput> & (type extends "inputs" ? EzInput [] : EzOutput[]) }
|
|
// */
|
|
// #getSlotItems(type) {
|
|
// // @ts-expect-error : these items are correct
|
|
// return (this.node[type] ?? []).reduce((p, s, i) => {
|
|
// if (s.name in p) {
|
|
// throw new Error(`Unable to store input ${s.name} on array as name conflicts.`);
|
|
// }
|
|
// // @ts-expect-error
|
|
// p.push((p[s.name] = new (type === "inputs" ? EzInput : EzOutput)(this, i, s)));
|
|
// return p;
|
|
// }, Object.assign([], { $: this }));
|
|
// }
|
|
|
|
/**
|
|
* @template { { new(node: EzNode, index: number, obj: any): any } } T
|
|
* @param { "inputs" | "outputs" | "widgets" | (() => Array<unknown>) } nodeProperty
|
|
* @param { string } nameProperty
|
|
* @param { T } ctor
|
|
* @returns { Record<string, InstanceType<T>> & Array<InstanceType<T>> }
|
|
*/
|
|
#makeLookupArray(nodeProperty, nameProperty, ctor) {
|
|
const items =
|
|
typeof nodeProperty === 'function'
|
|
? nodeProperty()
|
|
: this.node[nodeProperty]
|
|
return (items ?? []).reduce(
|
|
(p, s, i) => {
|
|
if (!s) return p
|
|
|
|
const name = s[nameProperty]
|
|
const item = new ctor(this, i, s)
|
|
p.push(item)
|
|
if (name) {
|
|
if (name in p) {
|
|
throw new Error(
|
|
`Unable to store ${nodeProperty} ${name} on array as name conflicts.`
|
|
)
|
|
}
|
|
}
|
|
p[name] = item
|
|
return p
|
|
},
|
|
Object.assign([], { $: this })
|
|
)
|
|
}
|
|
}
|
|
|
|
export class EzGraph {
|
|
/** @type { app } */
|
|
app
|
|
|
|
/**
|
|
* @param { app } app
|
|
*/
|
|
constructor(app) {
|
|
this.app = app
|
|
}
|
|
|
|
get nodes() {
|
|
return this.app.graph._nodes.map((n) => new EzNode(this.app, n))
|
|
}
|
|
|
|
clear() {
|
|
this.app.graph.clear()
|
|
}
|
|
|
|
arrange() {
|
|
this.app.graph.arrange()
|
|
}
|
|
|
|
stringify() {
|
|
return JSON.stringify(this.app.graph.serialize(), undefined)
|
|
}
|
|
|
|
/**
|
|
* @param { number | LGNode | EzNode } obj
|
|
* @returns { EzNode }
|
|
*/
|
|
find(obj) {
|
|
let match
|
|
let id
|
|
if (typeof obj === 'number') {
|
|
id = obj
|
|
} else {
|
|
id = obj.id
|
|
}
|
|
|
|
match = this.app.graph.getNodeById(id)
|
|
|
|
if (!match) {
|
|
throw new Error(`Unable to find node with ID ${id}.`)
|
|
}
|
|
|
|
return new EzNode(this.app, match)
|
|
}
|
|
|
|
/**
|
|
* @returns { Promise<void> }
|
|
*/
|
|
reload() {
|
|
const graph = JSON.parse(JSON.stringify(this.app.graph.serialize()))
|
|
return new Promise((r) => {
|
|
this.app.graph.clear()
|
|
setTimeout(async () => {
|
|
await this.app.loadGraphData(graph)
|
|
// @ts-expect-error
|
|
r()
|
|
}, 10)
|
|
})
|
|
}
|
|
|
|
/**
|
|
* @returns { Promise<{
|
|
* workflow: {},
|
|
* output: Record<string, {
|
|
* class_name: string,
|
|
* inputs: Record<string, [string, number] | unknown>
|
|
* }>}> }
|
|
*/
|
|
toPrompt() {
|
|
return this.app.graphToPrompt()
|
|
}
|
|
}
|
|
|
|
export const Ez = {
|
|
/**
|
|
* Quickly build and interact with a ComfyUI graph
|
|
* @example
|
|
* const { ez, graph } = Ez.graph(app);
|
|
* graph.clear();
|
|
* const [model, clip, vae] = ez.CheckpointLoaderSimple().outputs;
|
|
* const [pos] = ez.CLIPTextEncode(clip, { text: "positive" }).outputs;
|
|
* const [neg] = ez.CLIPTextEncode(clip, { text: "negative" }).outputs;
|
|
* const [latent] = ez.KSampler(model, pos, neg, ...ez.EmptyLatentImage().outputs).outputs;
|
|
* const [image] = ez.VAEDecode(latent, vae).outputs;
|
|
* const saveNode = ez.SaveImage(image);
|
|
* console.log(saveNode);
|
|
* graph.arrange();
|
|
* @param { app } app
|
|
* @param { boolean } clearGraph
|
|
* @param { LG["LiteGraph"] } LiteGraph
|
|
* @param { LG["LGraphCanvas"] } LGraphCanvas
|
|
* @returns { { graph: EzGraph, ez: Record<string, EzNodeFactory> } }
|
|
*/
|
|
graph(app, LiteGraph, LGraphCanvas, clearGraph = true) {
|
|
// Always set the active canvas so things work
|
|
LGraphCanvas.active_canvas = app.canvas
|
|
|
|
if (clearGraph) {
|
|
app.graph.clear()
|
|
}
|
|
|
|
const factory = new Proxy(
|
|
{},
|
|
{
|
|
get(_, p) {
|
|
if (typeof p !== 'string') throw new Error('Invalid node')
|
|
const node = LiteGraph.createNode(p)
|
|
if (!node) throw new Error(`Unknown node "${p}"`)
|
|
app.graph.add(node)
|
|
|
|
/**
|
|
* @param {Parameters<EzNodeFactory>} args
|
|
*/
|
|
return function (...args) {
|
|
const ezNode = new EzNode(app, node)
|
|
const inputs = ezNode.inputs
|
|
|
|
let slot = 0
|
|
for (const arg of args) {
|
|
if (arg instanceof EzOutput) {
|
|
arg.connectTo(inputs[slot++])
|
|
} else {
|
|
for (const k in arg) {
|
|
ezNode.widgets[k].value = arg[k]
|
|
}
|
|
}
|
|
}
|
|
|
|
return ezNode
|
|
}
|
|
}
|
|
}
|
|
)
|
|
|
|
return { graph: new EzGraph(app), ez: factory }
|
|
}
|
|
}
|