Spaces:
Running
on
Zero
Running
on
Zero
import { app } from "scripts/app.js"; | |
import type { | |
LLink, | |
LGraph, | |
ContextMenuItem, | |
LGraphCanvas, | |
SerializedLGraphNode, | |
LGraphNode as TLGraphNode, | |
IContextMenuOptions, | |
ContextMenu, | |
} from "typings/litegraph.js"; | |
import { addConnectionLayoutSupport } from "./utils.js"; | |
import { wait } from "rgthree/common/shared_utils.js"; | |
import { ComfyWidgets } from "scripts/widgets.js"; | |
import { BaseCollectorNode } from "./base_node_collector.js"; | |
import { NodeTypesString } from "./constants.js"; | |
/** | |
* The Collector Node. Takes any number of inputs as connections for nodes and collects them into | |
* one outputs. The next node will decide what to do with them. | |
* | |
* Currently only works with the Fast Muter, Fast Bypasser, and Fast Actions Button. | |
*/ | |
class CollectorNode extends BaseCollectorNode { | |
static override type = NodeTypesString.NODE_COLLECTOR; | |
static override title = NodeTypesString.NODE_COLLECTOR; | |
override comfyClass = NodeTypesString.NODE_COLLECTOR; | |
constructor(title = CollectorNode.title) { | |
super(title); | |
this.onConstructed(); | |
} | |
override onConstructed(): boolean { | |
this.addOutput("Output", "*"); | |
return super.onConstructed(); | |
} | |
override configure(info: SerializedLGraphNode<TLGraphNode>): void { | |
// Patch a small issue (~14h) where multiple OPT_CONNECTIONS may have been created. | |
// https://github.com/rgthree/rgthree-comfy/issues/206 | |
// TODO: This can probably be removed within a few weeks. | |
if (info.outputs?.length) { | |
info.outputs.length = 1; | |
} | |
super.configure(info); | |
} | |
} | |
/** Legacy "Combiner" */ | |
class CombinerNode extends CollectorNode { | |
static legacyType = "Node Combiner (rgthree)"; | |
static override title = "‼️ Node Combiner [DEPRECATED]"; | |
constructor(title = CombinerNode.title) { | |
super(title); | |
const note = ComfyWidgets["STRING"]( | |
this, | |
"last_seed", | |
["STRING", { multiline: true }], | |
app, | |
).widget; | |
note.inputEl!.value = | |
'The Node Combiner has been renamed to Node Collector. You can right-click and select "Update to Node Collector" to attempt to automatically update.'; | |
note.inputEl!.readOnly = true; | |
note.inputEl!.style.backgroundColor = "#332222"; | |
note.inputEl!.style.fontWeight = "bold"; | |
note.inputEl!.style.fontStyle = "italic"; | |
note.inputEl!.style.opacity = "0.8"; | |
this.getExtraMenuOptions = (_: LGraphCanvas, options: ContextMenuItem[]) => { | |
options.splice(options.length - 1, 0, { | |
content: "‼️ Update to Node Collector", | |
callback: ( | |
_value: ContextMenuItem, | |
_options: IContextMenuOptions, | |
_event: MouseEvent, | |
_parentMenu: ContextMenu | undefined, | |
_node: TLGraphNode, | |
) => { | |
updateCombinerToCollector(this); | |
}, | |
}); | |
}; | |
} | |
override configure(info: SerializedLGraphNode) { | |
super.configure(info); | |
if (this.title != CombinerNode.title && !this.title.startsWith("‼️")) { | |
this.title = "‼️ " + this.title; | |
} | |
} | |
} | |
/** | |
* Updates a Node Combiner to a Node Collector. | |
*/ | |
async function updateCombinerToCollector(node: TLGraphNode) { | |
if (node.type === CombinerNode.legacyType) { | |
// Create a new CollectorNode. | |
const newNode = new CollectorNode(); | |
if (node.title != CombinerNode.title) { | |
newNode.title = node.title.replace("‼️ ", ""); | |
} | |
// Port the position, size, and properties from the old node. | |
newNode.pos = [...node.pos]; | |
newNode.size = [...node.size]; | |
newNode.properties = { ...node.properties }; | |
// We now collect the links data, inputs and outputs, of the old node since these will be | |
// lost when we remove it. | |
const links: any[] = []; | |
for (const [index, output] of node.outputs.entries()) { | |
for (const linkId of output.links || []) { | |
const link: LLink = (app.graph as LGraph).links[linkId]!; | |
if (!link) continue; | |
const targetNode = app.graph.getNodeById(link.target_id); | |
links.push({ node: newNode, slot: index, targetNode, targetSlot: link.target_slot }); | |
} | |
} | |
for (const [index, input] of node.inputs.entries()) { | |
const linkId = input.link; | |
if (linkId) { | |
const link: LLink = (app.graph as LGraph).links[linkId]!; | |
const originNode = app.graph.getNodeById(link.origin_id); | |
links.push({ | |
node: originNode, | |
slot: link.origin_slot, | |
targetNode: newNode, | |
targetSlot: index, | |
}); | |
} | |
} | |
// Add the new node, remove the old node. | |
app.graph.add(newNode); | |
await wait(); | |
// Now go through and connect the other nodes up as they were. | |
for (const link of links) { | |
link.node.connect(link.slot, link.targetNode, link.targetSlot); | |
} | |
await wait(); | |
app.graph.remove(node); | |
} | |
} | |
app.registerExtension({ | |
name: "rgthree.NodeCollector", | |
registerCustomNodes() { | |
addConnectionLayoutSupport(CollectorNode, app, [ | |
["Left", "Right"], | |
["Right", "Left"], | |
]); | |
LiteGraph.registerNodeType(CollectorNode.title, CollectorNode); | |
CollectorNode.category = CollectorNode._category; | |
}, | |
}); | |
app.registerExtension({ | |
name: "rgthree.NodeCombiner", | |
registerCustomNodes() { | |
addConnectionLayoutSupport(CombinerNode, app, [ | |
["Left", "Right"], | |
["Right", "Left"], | |
]); | |
LiteGraph.registerNodeType(CombinerNode.legacyType, CombinerNode); | |
CombinerNode.category = CombinerNode._category; | |
}, | |
}); | |