| import { app } from "../../scripts/app.js"; |
| import { mergeIfValid, getWidgetConfig, setWidgetConfig } from "./widgetInputs.js"; |
|
|
| |
|
|
| app.registerExtension({ |
| name: "Comfy.RerouteNode", |
| registerCustomNodes(app) { |
| class RerouteNode { |
| constructor() { |
| if (!this.properties) { |
| this.properties = {}; |
| } |
| this.properties.showOutputText = RerouteNode.defaultVisibility; |
| this.properties.horizontal = false; |
|
|
| this.addInput("", "*"); |
| this.addOutput(this.properties.showOutputText ? "*" : "", "*"); |
|
|
| this.onAfterGraphConfigured = function () { |
| requestAnimationFrame(() => { |
| this.onConnectionsChange(LiteGraph.INPUT, null, true, null); |
| }); |
| }; |
|
|
| this.onConnectionsChange = function (type, index, connected, link_info) { |
| this.applyOrientation(); |
|
|
| |
| if (connected && type === LiteGraph.OUTPUT) { |
| |
| const types = new Set(this.outputs[0].links.map((l) => app.graph.links[l].type).filter((t) => t !== "*")); |
| if (types.size > 1) { |
| const linksToDisconnect = []; |
| for (let i = 0; i < this.outputs[0].links.length - 1; i++) { |
| const linkId = this.outputs[0].links[i]; |
| const link = app.graph.links[linkId]; |
| linksToDisconnect.push(link); |
| } |
| for (const link of linksToDisconnect) { |
| const node = app.graph.getNodeById(link.target_id); |
| node.disconnectInput(link.target_slot); |
| } |
| } |
| } |
|
|
| |
| let currentNode = this; |
| let updateNodes = []; |
| let inputType = null; |
| let inputNode = null; |
| while (currentNode) { |
| updateNodes.unshift(currentNode); |
| const linkId = currentNode.inputs[0].link; |
| if (linkId !== null) { |
| const link = app.graph.links[linkId]; |
| if (!link) return; |
| const node = app.graph.getNodeById(link.origin_id); |
| const type = node.constructor.type; |
| if (type === "Reroute") { |
| if (node === this) { |
| |
| currentNode.disconnectInput(link.target_slot); |
| currentNode = null; |
| } else { |
| |
| currentNode = node; |
| } |
| } else { |
| |
| inputNode = currentNode; |
| inputType = node.outputs[link.origin_slot]?.type ?? null; |
| break; |
| } |
| } else { |
| |
| currentNode = null; |
| break; |
| } |
| } |
|
|
| |
| const nodes = [this]; |
| let outputType = null; |
| while (nodes.length) { |
| currentNode = nodes.pop(); |
| const outputs = (currentNode.outputs ? currentNode.outputs[0].links : []) || []; |
| if (outputs.length) { |
| for (const linkId of outputs) { |
| const link = app.graph.links[linkId]; |
|
|
| |
| if (!link) continue; |
|
|
| const node = app.graph.getNodeById(link.target_id); |
| const type = node.constructor.type; |
|
|
| if (type === "Reroute") { |
| |
| nodes.push(node); |
| updateNodes.push(node); |
| } else { |
| |
| const nodeOutType = |
| node.inputs && node.inputs[link?.target_slot] && node.inputs[link.target_slot].type |
| ? node.inputs[link.target_slot].type |
| : null; |
| if (inputType && inputType !== "*" && nodeOutType !== inputType) { |
| |
| node.disconnectInput(link.target_slot); |
| } else { |
| outputType = nodeOutType; |
| } |
| } |
| } |
| } else { |
| |
| } |
| } |
|
|
| const displayType = inputType || outputType || "*"; |
| const color = LGraphCanvas.link_type_colors[displayType]; |
|
|
| let widgetConfig; |
| let targetWidget; |
| let widgetType; |
| |
| for (const node of updateNodes) { |
| |
| |
| node.outputs[0].type = inputType || "*"; |
| node.__outputType = displayType; |
| node.outputs[0].name = node.properties.showOutputText ? displayType : ""; |
| node.size = node.computeSize(); |
| node.applyOrientation(); |
|
|
| for (const l of node.outputs[0].links || []) { |
| const link = app.graph.links[l]; |
| if (link) { |
| link.color = color; |
|
|
| if (app.configuringGraph) continue; |
| const targetNode = app.graph.getNodeById(link.target_id); |
| const targetInput = targetNode.inputs?.[link.target_slot]; |
| if (targetInput?.widget) { |
| const config = getWidgetConfig(targetInput); |
| if (!widgetConfig) { |
| widgetConfig = config[1] ?? {}; |
| widgetType = config[0]; |
| } |
| if (!targetWidget) { |
| targetWidget = targetNode.widgets?.find((w) => w.name === targetInput.widget.name); |
| } |
|
|
| const merged = mergeIfValid(targetInput, [config[0], widgetConfig]); |
| if (merged.customConfig) { |
| widgetConfig = merged.customConfig; |
| } |
| } |
| } |
| } |
| } |
|
|
| for (const node of updateNodes) { |
| if (widgetConfig && outputType) { |
| node.inputs[0].widget = { name: "value" }; |
| setWidgetConfig(node.inputs[0], [widgetType ?? displayType, widgetConfig], targetWidget); |
| } else { |
| setWidgetConfig(node.inputs[0], null); |
| } |
| } |
|
|
| if (inputNode) { |
| const link = app.graph.links[inputNode.inputs[0].link]; |
| if (link) { |
| link.color = color; |
| } |
| } |
| }; |
|
|
| this.clone = function () { |
| const cloned = RerouteNode.prototype.clone.apply(this); |
| cloned.removeOutput(0); |
| cloned.addOutput(this.properties.showOutputText ? "*" : "", "*"); |
| cloned.size = cloned.computeSize(); |
| return cloned; |
| }; |
|
|
| |
| this.isVirtualNode = true; |
| } |
|
|
| getExtraMenuOptions(_, options) { |
| options.unshift( |
| { |
| content: (this.properties.showOutputText ? "Hide" : "Show") + " Type", |
| callback: () => { |
| this.properties.showOutputText = !this.properties.showOutputText; |
| if (this.properties.showOutputText) { |
| this.outputs[0].name = this.__outputType || this.outputs[0].type; |
| } else { |
| this.outputs[0].name = ""; |
| } |
| this.size = this.computeSize(); |
| this.applyOrientation(); |
| app.graph.setDirtyCanvas(true, true); |
| }, |
| }, |
| { |
| content: (RerouteNode.defaultVisibility ? "Hide" : "Show") + " Type By Default", |
| callback: () => { |
| RerouteNode.setDefaultTextVisibility(!RerouteNode.defaultVisibility); |
| }, |
| }, |
| { |
| |
| |
| |
| |
| content: "Set " + (this.properties.horizontal ? "Horizontal" : "Vertical"), |
| callback: () => { |
| this.properties.horizontal = !this.properties.horizontal; |
| this.applyOrientation(); |
| }, |
| } |
| ); |
| } |
| applyOrientation() { |
| this.horizontal = this.properties.horizontal; |
| if (this.horizontal) { |
| |
| |
| |
| this.inputs[0].pos = [this.size[0] / 2, 0]; |
| } else { |
| delete this.inputs[0].pos; |
| } |
| app.graph.setDirtyCanvas(true, true); |
| } |
|
|
| computeSize() { |
| return [ |
| this.properties.showOutputText && this.outputs && this.outputs.length |
| ? Math.max(75, LiteGraph.NODE_TEXT_SIZE * this.outputs[0].name.length * 0.6 + 40) |
| : 75, |
| 26, |
| ]; |
| } |
|
|
| static setDefaultTextVisibility(visible) { |
| RerouteNode.defaultVisibility = visible; |
| if (visible) { |
| localStorage["Comfy.RerouteNode.DefaultVisibility"] = "true"; |
| } else { |
| delete localStorage["Comfy.RerouteNode.DefaultVisibility"]; |
| } |
| } |
| } |
|
|
| |
| RerouteNode.setDefaultTextVisibility(!!localStorage["Comfy.RerouteNode.DefaultVisibility"]); |
|
|
| LiteGraph.registerNodeType( |
| "Reroute", |
| Object.assign(RerouteNode, { |
| title_mode: LiteGraph.NO_TITLE, |
| title: "Reroute", |
| collapsable: false, |
| }) |
| ); |
|
|
| RerouteNode.category = "utils"; |
| }, |
| }); |
|
|