multimodalart's picture
Squashing commit
4450790 verified
var _a, _b;
import { app } from "../../scripts/app.js";
import { getWidgetConfig, mergeIfValid, setWidgetConfig, } from "../../extensions/core/widgetInputs.js";
import { rgthreeConfig } from "../../rgthree/config.js";
import { rgthree } from "./rgthree.js";
import { IoDirection, LAYOUT_CLOCKWISE, LAYOUT_LABEL_OPPOSITES, LAYOUT_LABEL_TO_DATA, addConnectionLayoutSupport, addMenuItem, getSlotLinks, isValidConnection, setConnectionsLayout, waitForCanvas, } from "./utils.js";
import { SERVICE as KEY_EVENT_SERVICE } from "./services/key_events_services.js";
import { wait } from "../../rgthree/common/shared_utils.js";
import { RgthreeBaseVirtualNode } from "./base_node.js";
import { NodeTypesString } from "./constants.js";
const CONFIG_REROUTE = ((_a = rgthreeConfig === null || rgthreeConfig === void 0 ? void 0 : rgthreeConfig["nodes"]) === null || _a === void 0 ? void 0 : _a["reroute"]) || {};
const CONFIG_FAST_REROUTE = CONFIG_REROUTE["fast_reroute"];
const CONFIG_FAST_REROUTE_ENABLED = (_b = CONFIG_FAST_REROUTE["enabled"]) !== null && _b !== void 0 ? _b : false;
const CONFIG_KEY_CREATE_WHILE_LINKING = CONFIG_FAST_REROUTE["key_create_while_dragging_link"];
const CONFIG_KEY_ROTATE = CONFIG_FAST_REROUTE["key_rotate"];
const CONFIG_KEY_RESIZE = CONFIG_FAST_REROUTE["key_resize"];
const CONFIG_KEY_MOVE = CONFIG_FAST_REROUTE["key_move"];
const CONFIG_KEY_CXN_INPUT = CONFIG_FAST_REROUTE["key_connections_input"];
const CONFIG_KEY_CXN_OUTPUT = CONFIG_FAST_REROUTE["key_connections_output"];
let configWidth = Math.max(Math.round((Number(CONFIG_REROUTE["default_width"]) || 40) / 10) * 10, 10);
let configHeight = Math.max(Math.round((Number(CONFIG_REROUTE["default_height"]) || 30) / 10) * 10, 10);
while (configWidth * configHeight < 400) {
configWidth += 10;
configHeight += 10;
}
const configDefaultSize = [configWidth, configHeight];
const configResizable = !!CONFIG_REROUTE["default_resizable"];
let configLayout = CONFIG_REROUTE["default_layout"];
if (!Array.isArray(configLayout)) {
configLayout = ["Left", "Right"];
}
if (!LAYOUT_LABEL_TO_DATA[configLayout[0]]) {
configLayout[0] = "Left";
}
if (!LAYOUT_LABEL_TO_DATA[configLayout[1]] || configLayout[0] == configLayout[1]) {
configLayout[1] = LAYOUT_LABEL_OPPOSITES[configLayout[0]];
}
class RerouteService {
constructor() {
this.isFastLinking = false;
this.handledNewRerouteKeypress = false;
this.connectingData = null;
this.fastReroutesHistory = [];
this.handleLinkingKeydownBound = this.handleLinkingKeydown.bind(this);
this.handleLinkingKeyupBound = this.handleLinkingKeyup.bind(this);
if (CONFIG_FAST_REROUTE_ENABLED && (CONFIG_KEY_CREATE_WHILE_LINKING === null || CONFIG_KEY_CREATE_WHILE_LINKING === void 0 ? void 0 : CONFIG_KEY_CREATE_WHILE_LINKING.trim())) {
this.onCanvasSetUpListenerForLinking();
}
}
async onCanvasSetUpListenerForLinking() {
const canvas = await waitForCanvas();
const canvasProperty = true ? 'connecting_links' : 'connecting_node';
canvas[`_${canvasProperty}`];
const thisService = this;
Object.defineProperty(canvas, canvasProperty, {
get: function () {
return this[`_${canvasProperty}`];
},
set: function (value) {
var _a;
const isValNull = !value || !(value === null || value === void 0 ? void 0 : value.length);
const isPropNull = !this[`_${canvasProperty}`] || !((_a = this[`_${canvasProperty}`]) === null || _a === void 0 ? void 0 : _a.length);
const isStartingLinking = !isValNull && isPropNull;
const isStoppingLinking = !isPropNull && isValNull;
this[`_${canvasProperty}`] = value;
if (isStartingLinking) {
thisService.startingLinking();
}
if (isStoppingLinking) {
thisService.stoppingLinking();
thisService.connectingData = null;
}
},
});
}
startingLinking() {
this.isFastLinking = true;
KEY_EVENT_SERVICE.addEventListener("keydown", this.handleLinkingKeydownBound);
KEY_EVENT_SERVICE.addEventListener("keyup", this.handleLinkingKeyupBound);
}
stoppingLinking() {
this.isFastLinking = false;
this.fastReroutesHistory = [];
KEY_EVENT_SERVICE.removeEventListener("keydown", this.handleLinkingKeydownBound);
KEY_EVENT_SERVICE.removeEventListener("keyup", this.handleLinkingKeyupBound);
}
handleLinkingKeydown(event) {
if (!this.handledNewRerouteKeypress &&
KEY_EVENT_SERVICE.areOnlyKeysDown(CONFIG_KEY_CREATE_WHILE_LINKING)) {
this.handledNewRerouteKeypress = true;
this.insertNewRerouteWhileLinking();
}
}
handleLinkingKeyup(event) {
if (this.handledNewRerouteKeypress &&
!KEY_EVENT_SERVICE.areOnlyKeysDown(CONFIG_KEY_CREATE_WHILE_LINKING)) {
this.handledNewRerouteKeypress = false;
}
}
getConnectingData() {
var _a, _b;
const oldCanvas = app.canvas;
if (oldCanvas.connecting_node && oldCanvas.connecting_slot != null && ((_a = oldCanvas.connecting_pos) === null || _a === void 0 ? void 0 : _a.length)) {
return {
node: oldCanvas.connecting_node,
input: oldCanvas.connecting_input,
output: oldCanvas.connecting_output,
slot: oldCanvas.connecting_slot,
pos: [...oldCanvas.connecting_pos],
};
}
const canvas = app.canvas;
if ((_b = canvas.connecting_links) === null || _b === void 0 ? void 0 : _b.length) {
const link = canvas.connecting_links[0];
return {
node: link.node,
input: link.input,
output: link.output,
slot: link.slot,
pos: [...link.pos],
};
}
throw new Error("Error, handling linking keydown, but there's no link.");
}
setCanvasConnectingData(ctx) {
var _a, _b;
const oldCanvas = app.canvas;
if (oldCanvas.connecting_node && oldCanvas.connecting_slot != null && ((_a = oldCanvas.connecting_pos) === null || _a === void 0 ? void 0 : _a.length)) {
oldCanvas.connecting_node = ctx.node;
oldCanvas.connecting_input = ctx.input;
oldCanvas.connecting_output = ctx.output;
oldCanvas.connecting_slot = ctx.slot;
oldCanvas.connecting_pos = ctx.pos;
}
const canvas = app.canvas;
if ((_b = canvas.connecting_links) === null || _b === void 0 ? void 0 : _b.length) {
const link = canvas.connecting_links[0];
link.node = ctx.node;
link.input = ctx.input;
link.output = ctx.output;
link.slot = ctx.slot;
link.pos = ctx.pos;
}
}
insertNewRerouteWhileLinking() {
var _a;
const canvas = app.canvas;
this.connectingData = this.getConnectingData();
if (!this.connectingData) {
throw new Error("Error, handling linking keydown, but there's no link.");
}
const data = this.connectingData;
const node = LiteGraph.createNode("Reroute (rgthree)");
const entry = {
node,
previous: { ...this.connectingData },
current: undefined,
};
this.fastReroutesHistory.push(entry);
let connectingDir = (_a = (data.input || data.output)) === null || _a === void 0 ? void 0 : _a.dir;
if (!connectingDir) {
connectingDir = data.input ? LiteGraph.LEFT : LiteGraph.RIGHT;
}
let newPos = canvas.convertEventToCanvasOffset({
clientX: Math.round(canvas.last_mouse_position[0] / 10) * 10,
clientY: Math.round(canvas.last_mouse_position[1] / 10) * 10,
});
entry.node.pos = newPos;
canvas.graph.add(entry.node);
canvas.selectNode(entry.node);
const distX = entry.node.pos[0] - data.pos[0];
const distY = entry.node.pos[1] - data.pos[1];
const layout = ["Left", "Right"];
if (distX > 0 && Math.abs(distX) > Math.abs(distY)) {
layout[0] = data.output ? "Left" : "Right";
layout[1] = LAYOUT_LABEL_OPPOSITES[layout[0]];
node.pos[0] -= node.size[0] + 10;
node.pos[1] -= Math.round(node.size[1] / 2 / 10) * 10;
}
else if (distX < 0 && Math.abs(distX) > Math.abs(distY)) {
layout[0] = data.output ? "Right" : "Left";
layout[1] = LAYOUT_LABEL_OPPOSITES[layout[0]];
node.pos[1] -= Math.round(node.size[1] / 2 / 10) * 10;
}
else if (distY < 0 && Math.abs(distY) > Math.abs(distX)) {
layout[0] = data.output ? "Bottom" : "Top";
layout[1] = LAYOUT_LABEL_OPPOSITES[layout[0]];
node.pos[0] -= Math.round(node.size[0] / 2 / 10) * 10;
}
else if (distY > 0 && Math.abs(distY) > Math.abs(distX)) {
layout[0] = data.output ? "Top" : "Bottom";
layout[1] = LAYOUT_LABEL_OPPOSITES[layout[0]];
node.pos[0] -= Math.round(node.size[0] / 2 / 10) * 10;
node.pos[1] -= node.size[1] + 10;
}
setConnectionsLayout(entry.node, layout);
if (data.output) {
data.node.connect(data.slot, entry.node, 0);
data.node = entry.node;
data.output = entry.node.outputs[0];
data.slot = 0;
data.pos = entry.node.getConnectionPos(false, 0);
}
else {
entry.node.connect(0, data.node, data.slot);
data.node = entry.node;
data.input = entry.node.inputs[0];
data.slot = 0;
data.pos = entry.node.getConnectionPos(true, 0);
}
this.setCanvasConnectingData(data);
entry.current = { ...this.connectingData };
app.graph.setDirtyCanvas(true, true);
}
handleMoveOrResizeNodeMaybeWhileDragging(node) {
const data = this.connectingData;
if (this.isFastLinking && node === (data === null || data === void 0 ? void 0 : data.node)) {
const entry = this.fastReroutesHistory[this.fastReroutesHistory.length - 1];
if (entry) {
data.pos = entry.node.getConnectionPos(!!data.input, 0);
this.setCanvasConnectingData(data);
}
}
}
handleRemovedNodeMaybeWhileDragging(node) {
const currentEntry = this.fastReroutesHistory[this.fastReroutesHistory.length - 1];
if ((currentEntry === null || currentEntry === void 0 ? void 0 : currentEntry.node) === node) {
this.setCanvasConnectingData(currentEntry.previous);
this.fastReroutesHistory.splice(this.fastReroutesHistory.length - 1, 1);
if (currentEntry.previous.node) {
app.canvas.selectNode(currentEntry.previous.node);
}
}
}
}
const SERVICE = new RerouteService();
class RerouteNode extends RgthreeBaseVirtualNode {
constructor(title = RerouteNode.title) {
super(title);
this.comfyClass = NodeTypesString.REROUTE;
this.isVirtualNode = true;
this.hideSlotLabels = true;
this.schedulePromise = null;
this.defaultConnectionsLayout = Array.from(configLayout);
this.shortcuts = {
rotate: { keys: CONFIG_KEY_ROTATE, state: false },
connection_input: { keys: CONFIG_KEY_CXN_INPUT, state: false },
connection_output: { keys: CONFIG_KEY_CXN_OUTPUT, state: false },
resize: {
keys: CONFIG_KEY_RESIZE,
state: false,
initialMousePos: [-1, -1],
initialNodeSize: [-1, -1],
initialNodePos: [-1, -1],
resizeOnSide: [-1, -1],
},
move: {
keys: CONFIG_KEY_MOVE,
state: false,
initialMousePos: [-1, -1],
initialNodePos: [-1, -1],
},
};
this.onConstructed();
}
onConstructed() {
var _a;
this.setResizable((_a = this.properties["resizable"]) !== null && _a !== void 0 ? _a : configResizable);
this.size = RerouteNode.size;
this.addInput("", "*");
this.addOutput("", "*");
setTimeout(() => this.applyNodeSize(), 20);
return super.onConstructed();
}
configure(info) {
var _a, _b, _c;
if ((_a = info.outputs) === null || _a === void 0 ? void 0 : _a.length) {
info.outputs.length = 1;
}
if ((_b = info.inputs) === null || _b === void 0 ? void 0 : _b.length) {
info.inputs.length = 1;
}
super.configure(info);
this.configuring = true;
this.setResizable((_c = this.properties["resizable"]) !== null && _c !== void 0 ? _c : configResizable);
this.applyNodeSize();
this.configuring = false;
}
setResizable(resizable) {
this.properties["resizable"] = !!resizable;
this.resizable = this.properties["resizable"];
}
clone() {
const cloned = super.clone();
cloned.inputs[0].type = "*";
cloned.outputs[0].type = "*";
return cloned;
}
onConnectionsChange(type, _slotIndex, connected, _link_info, _ioSlot) {
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);
}
}
}
this.scheduleStabilize();
}
onDrawForeground(ctx, canvas) {
var _a, _b, _c, _d;
if ((_a = this.properties) === null || _a === void 0 ? void 0 : _a["showLabel"]) {
const low_quality = ((_b = canvas === null || canvas === void 0 ? void 0 : canvas.ds) === null || _b === void 0 ? void 0 : _b.scale) && canvas.ds.scale < 0.6;
if (low_quality || this.size[0] <= 10) {
return;
}
const fontSize = Math.min(14, (this.size[1] * 0.65) | 0);
ctx.save();
ctx.fillStyle = "#888";
ctx.font = `${fontSize}px Arial`;
ctx.textAlign = "center";
ctx.textBaseline = "middle";
ctx.fillText(String(this.title && this.title !== RerouteNode.title
? this.title
: ((_d = (_c = this.outputs) === null || _c === void 0 ? void 0 : _c[0]) === null || _d === void 0 ? void 0 : _d.type) || ""), this.size[0] / 2, this.size[1] / 2, this.size[0] - 30);
ctx.restore();
}
}
findInputSlot(name) {
return 0;
}
findOutputSlot(name) {
return 0;
}
disconnectOutput(slot, targetNode) {
return super.disconnectOutput(slot, targetNode);
}
disconnectInput(slot) {
var _a;
if (rgthree.replacingReroute != null && ((_a = this.inputs[0]) === null || _a === void 0 ? void 0 : _a.link)) {
const graph = app.graph;
const link = graph.links[this.inputs[0].link];
const node = graph.getNodeById(link === null || link === void 0 ? void 0 : link.origin_id);
if (rgthree.replacingReroute !== (node === null || node === void 0 ? void 0 : node.id)) {
return false;
}
}
return super.disconnectInput(slot);
}
scheduleStabilize(ms = 64) {
if (!this.schedulePromise) {
this.schedulePromise = new Promise((resolve) => {
setTimeout(() => {
this.schedulePromise = null;
this.stabilize();
resolve();
}, ms);
});
}
return this.schedulePromise;
}
stabilize() {
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l;
if (this.configuring) {
return;
}
let currentNode = this;
let updateNodes = [];
let input = null;
let inputType = null;
let inputNode = null;
let inputNodeOutputSlot = null;
while (currentNode) {
updateNodes.unshift(currentNode);
const linkId = currentNode.inputs[0].link;
if (linkId !== null) {
const link = app.graph.links[linkId];
const node = app.graph.getNodeById(link.origin_id);
if (!node) {
app.graph.removeLink(linkId);
currentNode = null;
break;
}
const type = node.constructor.type;
if (type === null || type === void 0 ? void 0 : type.includes("Reroute")) {
if (node === this) {
currentNode.disconnectInput(link.target_slot);
currentNode = null;
}
else {
currentNode = node;
}
}
else {
inputNode = node;
inputNodeOutputSlot = link.origin_slot;
input = (_a = node.outputs[inputNodeOutputSlot]) !== null && _a !== void 0 ? _a : null;
inputType = (_b = input === null || input === void 0 ? void 0 : input.type) !== null && _b !== void 0 ? _b : null;
break;
}
}
else {
currentNode = null;
break;
}
}
const nodes = [this];
let outputNode = null;
let outputType = null;
let outputWidgetConfig = null;
let outputWidget = 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);
if (!node)
continue;
const type = node.constructor.type;
if (type === null || type === void 0 ? void 0 : type.includes("Reroute")) {
nodes.push(node);
updateNodes.push(node);
}
else {
const output = (_d = (_c = node.inputs) === null || _c === void 0 ? void 0 : _c[link.target_slot]) !== null && _d !== void 0 ? _d : null;
const nodeOutType = output === null || output === void 0 ? void 0 : output.type;
if (nodeOutType == null) {
console.warn(`[rgthree] Reroute - Connected node ${node.id} does not have type information for ` +
`slot ${link.target_slot}. Skipping connection enforcement, but something is odd ` +
`with that node.`);
}
else if (inputType &&
inputType !== "*" &&
nodeOutType !== "*" &&
!isValidConnection(input, output)) {
console.warn(`[rgthree] Reroute - Disconnecting connected node's input (${node.id}.${link.target_slot}) (${node.type}) because its type (${String(nodeOutType)}) does not match the reroute type (${String(inputType)})`);
node.disconnectInput(link.target_slot);
}
else {
outputType = nodeOutType;
outputNode = node;
outputWidgetConfig = null;
outputWidget = null;
if (output === null || output === void 0 ? void 0 : output.widget) {
try {
const config = getWidgetConfig(output);
if (!outputWidgetConfig && config) {
outputWidgetConfig = (_e = config[1]) !== null && _e !== void 0 ? _e : {};
outputType = config[0];
if (!outputWidget) {
outputWidget = (_f = outputNode.widgets) === null || _f === void 0 ? void 0 : _f.find((w) => { var _a; return w.name === ((_a = output === null || output === void 0 ? void 0 : output.widget) === null || _a === void 0 ? void 0 : _a.name); });
}
const merged = mergeIfValid(output, [config[0], outputWidgetConfig]);
if (merged.customConfig) {
outputWidgetConfig = merged.customConfig;
}
}
}
catch (e) {
console.error("[rgthree] Could not propagate widget infor for reroute; maybe ComfyUI updated?");
outputWidgetConfig = null;
outputWidget = null;
}
}
}
}
}
}
else {
}
}
const displayType = inputType || outputType || "*";
const color = LGraphCanvas.link_type_colors[displayType];
for (const node of updateNodes) {
node.outputs[0].type = inputType || "*";
node.__outputType = displayType;
node.outputs[0].name = (input === null || input === void 0 ? void 0 : input.name) || "";
node.size = node.computeSize();
(_h = (_g = node).applyNodeSize) === null || _h === void 0 ? void 0 : _h.call(_g);
for (const l of node.outputs[0].links || []) {
const link = app.graph.links[l];
if (link) {
link.color = color;
}
}
try {
if (outputWidgetConfig && outputWidget && outputType) {
node.inputs[0].widget = { name: "value" };
setWidgetConfig(node.inputs[0], [outputType !== null && outputType !== void 0 ? outputType : displayType, outputWidgetConfig], outputWidget);
}
else {
setWidgetConfig(node.inputs[0], null);
}
}
catch (e) {
console.error("[rgthree] Could not set widget config for reroute; maybe ComfyUI updated?");
outputWidgetConfig = null;
outputWidget = null;
if ((_j = node.inputs[0]) === null || _j === void 0 ? void 0 : _j.widget) {
delete node.inputs[0].widget;
}
}
}
if (inputNode && inputNodeOutputSlot != null) {
const links = inputNode.outputs[inputNodeOutputSlot].links;
for (const l of links || []) {
const link = app.graph.links[l];
if (link) {
link.color = color;
}
}
}
(_k = inputNode === null || inputNode === void 0 ? void 0 : inputNode.onConnectionsChainChange) === null || _k === void 0 ? void 0 : _k.call(inputNode);
(_l = outputNode === null || outputNode === void 0 ? void 0 : outputNode.onConnectionsChainChange) === null || _l === void 0 ? void 0 : _l.call(outputNode);
app.graph.setDirtyCanvas(true, true);
}
setSize(size) {
const oldSize = [...this.size];
const newSize = [...size];
super.setSize(newSize);
this.properties["size"] = [...this.size];
this.stabilizeLayout(oldSize, newSize);
}
stabilizeLayout(oldSize, newSize) {
if (newSize[0] === 10 || newSize[1] === 10) {
const props = this.properties;
props["connections_layout"] = props["connections_layout"] || ["Left", "Right"];
const layout = props["connections_layout"];
props["connections_dir"] = props["connections_dir"] || [-1, -1];
const dir = props["connections_dir"];
if (oldSize[0] > 10 && newSize[0] === 10) {
dir[0] = LiteGraph.DOWN;
dir[1] = LiteGraph.UP;
if (layout[0] === "Bottom") {
layout[1] = "Top";
}
else if (layout[1] === "Top") {
layout[0] = "Bottom";
}
else {
layout[0] = "Top";
layout[1] = "Bottom";
dir[0] = LiteGraph.UP;
dir[1] = LiteGraph.DOWN;
}
this.setDirtyCanvas(true, true);
}
else if (oldSize[1] > 10 && newSize[1] === 10) {
dir[0] = LiteGraph.RIGHT;
dir[1] = LiteGraph.LEFT;
if (layout[0] === "Right") {
layout[1] = "Left";
}
else if (layout[1] === "Left") {
layout[0] = "Right";
}
else {
layout[0] = "Left";
layout[1] = "Right";
dir[0] = LiteGraph.LEFT;
dir[1] = LiteGraph.RIGHT;
}
this.setDirtyCanvas(true, true);
}
}
SERVICE.handleMoveOrResizeNodeMaybeWhileDragging(this);
}
applyNodeSize() {
this.properties["size"] = this.properties["size"] || RerouteNode.size;
this.properties["size"] = [
Number(this.properties["size"][0]),
Number(this.properties["size"][1]),
];
this.size = this.properties["size"];
app.graph.setDirtyCanvas(true, true);
}
rotate(degrees) {
const w = this.size[0];
const h = this.size[1];
this.properties["connections_layout"] =
this.properties["connections_layout"] || this.defaultConnectionsLayout;
const inputDirIndex = LAYOUT_CLOCKWISE.indexOf(this.properties["connections_layout"][0]);
const outputDirIndex = LAYOUT_CLOCKWISE.indexOf(this.properties["connections_layout"][1]);
if (degrees == 90 || degrees === -90) {
if (degrees === -90) {
this.properties["connections_layout"][0] =
LAYOUT_CLOCKWISE[(((inputDirIndex - 1) % 4) + 4) % 4];
this.properties["connections_layout"][1] =
LAYOUT_CLOCKWISE[(((outputDirIndex - 1) % 4) + 4) % 4];
}
else {
this.properties["connections_layout"][0] =
LAYOUT_CLOCKWISE[(((inputDirIndex + 1) % 4) + 4) % 4];
this.properties["connections_layout"][1] =
LAYOUT_CLOCKWISE[(((outputDirIndex + 1) % 4) + 4) % 4];
}
}
else if (degrees === 180) {
this.properties["connections_layout"][0] =
LAYOUT_CLOCKWISE[(((inputDirIndex + 2) % 4) + 4) % 4];
this.properties["connections_layout"][1] =
LAYOUT_CLOCKWISE[(((outputDirIndex + 2) % 4) + 4) % 4];
}
this.setSize([h, w]);
}
manuallyHandleMove(event) {
const shortcut = this.shortcuts.move;
if (shortcut.state) {
const diffX = Math.round((event.clientX - shortcut.initialMousePos[0]) / 10) * 10;
const diffY = Math.round((event.clientY - shortcut.initialMousePos[1]) / 10) * 10;
this.pos[0] = shortcut.initialNodePos[0] + diffX;
this.pos[1] = shortcut.initialNodePos[1] + diffY;
this.setDirtyCanvas(true, true);
SERVICE.handleMoveOrResizeNodeMaybeWhileDragging(this);
}
}
manuallyHandleResize(event) {
const shortcut = this.shortcuts.resize;
if (shortcut.state) {
let diffX = Math.round((event.clientX - shortcut.initialMousePos[0]) / 10) * 10;
let diffY = Math.round((event.clientY - shortcut.initialMousePos[1]) / 10) * 10;
diffX *= shortcut.resizeOnSide[0] === LiteGraph.LEFT ? -1 : 1;
diffY *= shortcut.resizeOnSide[1] === LiteGraph.UP ? -1 : 1;
const oldSize = [...this.size];
this.setSize([
Math.max(10, shortcut.initialNodeSize[0] + diffX),
Math.max(10, shortcut.initialNodeSize[1] + diffY),
]);
if (shortcut.resizeOnSide[0] === LiteGraph.LEFT && oldSize[0] > 10) {
this.pos[0] = shortcut.initialNodePos[0] - diffX;
}
if (shortcut.resizeOnSide[1] === LiteGraph.UP && oldSize[1] > 10) {
this.pos[1] = shortcut.initialNodePos[1] - diffY;
}
this.setDirtyCanvas(true, true);
}
}
cycleConnection(ioDir) {
var _a, _b;
const props = this.properties;
props["connections_layout"] = props["connections_layout"] || ["Left", "Right"];
const propIdx = ioDir == IoDirection.INPUT ? 0 : 1;
const oppositeIdx = propIdx ? 0 : 1;
let currentLayout = props["connections_layout"][propIdx];
let oppositeLayout = props["connections_layout"][oppositeIdx];
if (this.size[0] === 10 || this.size[1] === 10) {
props["connections_dir"] = props["connections_dir"] || [-1, -1];
let currentDir = props["connections_dir"][propIdx];
const options = this.size[0] === 10
? currentLayout === "Bottom"
? [LiteGraph.DOWN, LiteGraph.RIGHT, LiteGraph.LEFT]
: [LiteGraph.UP, LiteGraph.LEFT, LiteGraph.RIGHT]
: currentLayout === "Right"
? [LiteGraph.RIGHT, LiteGraph.DOWN, LiteGraph.UP]
: [LiteGraph.LEFT, LiteGraph.UP, LiteGraph.DOWN];
let idx = options.indexOf(currentDir);
let next = (_a = options[idx + 1]) !== null && _a !== void 0 ? _a : options[0];
this.properties["connections_dir"][propIdx] = next;
return;
}
let next = currentLayout;
do {
let idx = LAYOUT_CLOCKWISE.indexOf(next);
next = (_b = LAYOUT_CLOCKWISE[idx + 1]) !== null && _b !== void 0 ? _b : LAYOUT_CLOCKWISE[0];
} while (next === oppositeLayout);
this.properties["connections_layout"][propIdx] = next;
this.setDirtyCanvas(true, true);
}
onMouseMove(event) {
if (this.shortcuts.move.state) {
const shortcut = this.shortcuts.move;
if (shortcut.initialMousePos[0] === -1) {
shortcut.initialMousePos[0] = event.clientX;
shortcut.initialMousePos[1] = event.clientY;
shortcut.initialNodePos[0] = this.pos[0];
shortcut.initialNodePos[1] = this.pos[1];
}
this.manuallyHandleMove(event);
}
else if (this.shortcuts.resize.state) {
const shortcut = this.shortcuts.resize;
if (shortcut.initialMousePos[0] === -1) {
shortcut.initialMousePos[0] = event.clientX;
shortcut.initialMousePos[1] = event.clientY;
shortcut.initialNodeSize[0] = this.size[0];
shortcut.initialNodeSize[1] = this.size[1];
shortcut.initialNodePos[0] = this.pos[0];
shortcut.initialNodePos[1] = this.pos[1];
const canvas = app.canvas;
const offset = canvas.convertEventToCanvasOffset(event);
shortcut.resizeOnSide[0] = this.pos[0] > offset[0] ? LiteGraph.LEFT : LiteGraph.RIGHT;
shortcut.resizeOnSide[1] = this.pos[1] > offset[1] ? LiteGraph.UP : LiteGraph.DOWN;
}
this.manuallyHandleResize(event);
}
}
onKeyDown(event) {
super.onKeyDown(event);
const canvas = app.canvas;
if (CONFIG_FAST_REROUTE_ENABLED) {
for (const [key, shortcut] of Object.entries(this.shortcuts)) {
if (!shortcut.state) {
const keys = KEY_EVENT_SERVICE.areOnlyKeysDown(shortcut.keys);
if (keys) {
shortcut.state = true;
if (key === "rotate") {
this.rotate(90);
}
else if (key.includes("connection")) {
this.cycleConnection(key.includes("input") ? IoDirection.INPUT : IoDirection.OUTPUT);
}
if (shortcut.initialMousePos) {
canvas.node_capturing_input = this;
}
}
}
}
}
}
onKeyUp(event) {
super.onKeyUp(event);
const canvas = app.canvas;
if (CONFIG_FAST_REROUTE_ENABLED) {
for (const [key, shortcut] of Object.entries(this.shortcuts)) {
if (shortcut.state) {
const keys = KEY_EVENT_SERVICE.areOnlyKeysDown(shortcut.keys);
if (!keys) {
shortcut.state = false;
if (shortcut.initialMousePos) {
shortcut.initialMousePos = [-1, -1];
if ((canvas.node_capturing_input = this)) {
canvas.node_capturing_input = null;
}
this.setDirtyCanvas(true, true);
}
}
}
}
}
}
onDeselected() {
var _a;
(_a = super.onDeselected) === null || _a === void 0 ? void 0 : _a.call(this);
const canvas = app.canvas;
for (const [key, shortcut] of Object.entries(this.shortcuts)) {
shortcut.state = false;
if (shortcut.initialMousePos) {
shortcut.initialMousePos = [-1, -1];
if ((canvas.node_capturing_input = this)) {
canvas.node_capturing_input = null;
}
this.setDirtyCanvas(true, true);
}
}
}
onRemoved() {
var _a;
(_a = super.onRemoved) === null || _a === void 0 ? void 0 : _a.call(this);
setTimeout(() => {
SERVICE.handleRemovedNodeMaybeWhileDragging(this);
}, 32);
}
getHelp() {
return `
<p>
Finally, a comfortable, powerful reroute node with true multi-direction and powerful
shortcuts to bring your workflow to the next level.
</p>
${!CONFIG_FAST_REROUTE_ENABLED
? `<p><i>Fast Shortcuts are currently disabled.</b>`
: `
<ul>
<li><p>
<code>${CONFIG_KEY_CREATE_WHILE_LINKING}</code> Create a new reroute node while dragging
a link, connecting it to the link in the place and continuing the link.
</p></li>
<li><p>
<code>${CONFIG_KEY_ROTATE}</code> Rotate the selected reroute node counter clockwise 90
degrees.
</p></li>
<li><p>
<code>${CONFIG_KEY_RESIZE}</code> Resize the selected reroute node from the nearest
corner by holding down and moving your mouse.
</p></li>
<li><p>
<code>${CONFIG_KEY_MOVE}</code> Move the selected reroute node by holding down and
moving your mouse.
</p></li>
<li><p>
<code>${CONFIG_KEY_CXN_INPUT}</code> Change the input layout/direction of the selected
reroute node.
</p></li>
<li><p>
<code>${CONFIG_KEY_CXN_OUTPUT}</code> Change the output layout/direction of the selected
reroute node.
</p></li>
</ul>
`}
<p><small>
To change, ${!CONFIG_FAST_REROUTE_ENABLED ? "enable" : "disable"} or configure sohrtcuts,
make a copy of
<code>/custom_nodes/rgthree-comfy/rgthree_config.json.default</code> to
<code>/custom_nodes/rgthree-comfy/rgthree_config.json</code> and configure under
<code>nodes > reroute > fast_reroute</code>.
</small></p>
`;
}
}
RerouteNode.title = NodeTypesString.REROUTE;
RerouteNode.type = NodeTypesString.REROUTE;
RerouteNode.title_mode = LiteGraph.NO_TITLE;
RerouteNode.collapsable = false;
RerouteNode.layout_slot_offset = 5;
RerouteNode.size = configDefaultSize;
addMenuItem(RerouteNode, app, {
name: (node) => { var _a; return `${((_a = node.properties) === null || _a === void 0 ? void 0 : _a["showLabel"]) ? "Hide" : "Show"} Label/Title`; },
property: "showLabel",
callback: async (node, value) => {
app.graph.setDirtyCanvas(true, true);
},
});
addMenuItem(RerouteNode, app, {
name: (node) => `${node.resizable ? "No" : "Allow"} Resizing`,
callback: (node) => {
node.setResizable(!node.resizable);
node.size[0] = Math.max(40, node.size[0]);
node.size[1] = Math.max(30, node.size[1]);
node.applyNodeSize();
},
});
addMenuItem(RerouteNode, app, {
name: "Static Width",
property: "size",
subMenuOptions: (() => {
const options = [];
for (let w = 8; w > 0; w--) {
options.push(`${w * 10}`);
}
return options;
})(),
prepareValue: (value, node) => [Number(value), node.size[1]],
callback: (node) => {
node.setResizable(false);
node.applyNodeSize();
},
});
addMenuItem(RerouteNode, app, {
name: "Static Height",
property: "size",
subMenuOptions: (() => {
const options = [];
for (let w = 8; w > 0; w--) {
options.push(`${w * 10}`);
}
return options;
})(),
prepareValue: (value, node) => [node.size[0], Number(value)],
callback: (node) => {
node.setResizable(false);
node.applyNodeSize();
},
});
addConnectionLayoutSupport(RerouteNode, app, [
["Left", "Right"],
["Left", "Top"],
["Left", "Bottom"],
["Right", "Left"],
["Right", "Top"],
["Right", "Bottom"],
["Top", "Left"],
["Top", "Right"],
["Top", "Bottom"],
["Bottom", "Left"],
["Bottom", "Right"],
["Bottom", "Top"],
], (node) => {
node.applyNodeSize();
});
addMenuItem(RerouteNode, app, {
name: "Rotate",
subMenuOptions: [
"Rotate 90° Clockwise",
"Rotate 90° Counter-Clockwise",
"Rotate 180°",
null,
"Flip Horizontally",
"Flip Vertically",
],
callback: (node_, value) => {
const node = node_;
if (value === null || value === void 0 ? void 0 : value.startsWith("Rotate 90° Clockwise")) {
node.rotate(90);
}
else if (value === null || value === void 0 ? void 0 : value.startsWith("Rotate 90° Counter-Clockwise")) {
node.rotate(-90);
}
else if (value === null || value === void 0 ? void 0 : value.startsWith("Rotate 180°")) {
node.rotate(180);
}
else {
const inputDirIndex = LAYOUT_CLOCKWISE.indexOf(node.properties["connections_layout"][0]);
const outputDirIndex = LAYOUT_CLOCKWISE.indexOf(node.properties["connections_layout"][1]);
if (value === null || value === void 0 ? void 0 : value.startsWith("Flip Horizontally")) {
if (["Left", "Right"].includes(node.properties["connections_layout"][0])) {
node.properties["connections_layout"][0] =
LAYOUT_CLOCKWISE[(((inputDirIndex + 2) % 4) + 4) % 4];
}
if (["Left", "Right"].includes(node.properties["connections_layout"][1])) {
node.properties["connections_layout"][1] =
LAYOUT_CLOCKWISE[(((outputDirIndex + 2) % 4) + 4) % 4];
}
}
else if (value === null || value === void 0 ? void 0 : value.startsWith("Flip Vertically")) {
if (["Top", "Bottom"].includes(node.properties["connections_layout"][0])) {
node.properties["connections_layout"][0] =
LAYOUT_CLOCKWISE[(((inputDirIndex + 2) % 4) + 4) % 4];
}
if (["Top", "Bottom"].includes(node.properties["connections_layout"][1])) {
node.properties["connections_layout"][1] =
LAYOUT_CLOCKWISE[(((outputDirIndex + 2) % 4) + 4) % 4];
}
}
}
},
});
addMenuItem(RerouteNode, app, {
name: "Clone New Reroute...",
subMenuOptions: ["Before", "After"],
callback: async (node, value) => {
const clone = node.clone();
const pos = [...node.pos];
if (value === "Before") {
clone.pos = [pos[0] - 20, pos[1] - 20];
app.graph.add(clone);
await wait();
const inputLinks = getSlotLinks(node.inputs[0]);
for (const inputLink of inputLinks) {
const link = inputLink.link;
const linkedNode = app.graph.getNodeById(link.origin_id);
if (linkedNode) {
linkedNode.connect(0, clone, 0);
}
}
clone.connect(0, node, 0);
}
else {
clone.pos = [pos[0] + 20, pos[1] + 20];
app.graph.add(clone);
await wait();
const outputLinks = getSlotLinks(node.outputs[0]);
node.connect(0, clone, 0);
for (const outputLink of outputLinks) {
const link = outputLink.link;
const linkedNode = app.graph.getNodeById(link.target_id);
if (linkedNode) {
clone.connect(0, linkedNode, link.target_slot);
}
}
}
},
});
app.registerExtension({
name: "rgthree.Reroute",
registerCustomNodes() {
RerouteNode.setUp();
},
});