Some checks failed
Python Linting / Run Ruff (push) Has been cancelled
Python Linting / Run Pylint (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.10, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.11, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-stable (12.1, , linux, 3.12, [self-hosted Linux], stable) (push) Has been cancelled
Full Comfy CI Workflow Runs / test-unix-nightly (12.1, , linux, 3.11, [self-hosted Linux], nightly) (push) Has been cancelled
Execution Tests / test (macos-latest) (push) Has been cancelled
Execution Tests / test (ubuntu-latest) (push) Has been cancelled
Execution Tests / test (windows-latest) (push) Has been cancelled
Test server launches without errors / test (push) Has been cancelled
Unit Tests / test (macos-latest) (push) Has been cancelled
Unit Tests / test (ubuntu-latest) (push) Has been cancelled
Unit Tests / test (windows-2022) (push) Has been cancelled
Includes 30 custom nodes committed directly, 7 Civitai-exclusive loras stored via Git LFS, and a setup script that installs all dependencies and downloads HuggingFace-hosted models on vast.ai. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
360 lines
11 KiB
TypeScript
360 lines
11 KiB
TypeScript
import type {
|
||
LGraph,
|
||
LGraphNode,
|
||
ISerialisedNode,
|
||
IButtonWidget,
|
||
IComboWidget,
|
||
IWidget,
|
||
IBaseWidget,
|
||
} from "@comfyorg/frontend";
|
||
import type {ComfyApp} from "@comfyorg/frontend";
|
||
import type {RgthreeBaseVirtualNode} from "./base_node.js";
|
||
|
||
import {app} from "scripts/app.js";
|
||
import {BaseAnyInputConnectedNode} from "./base_any_input_connected_node.js";
|
||
import {NodeTypesString} from "./constants.js";
|
||
import {addMenuItem, changeModeOfNodes} from "./utils.js";
|
||
import {rgthree} from "./rgthree.js";
|
||
|
||
const MODE_ALWAYS = 0;
|
||
const MODE_MUTE = 2;
|
||
const MODE_BYPASS = 4;
|
||
|
||
/**
|
||
* The Fast Actions Button.
|
||
*
|
||
* This adds a button that the user can connect any node to and then choose an action to take on
|
||
* that node when the button is pressed. Default actions are "Mute," "Bypass," and "Enable," but
|
||
* Nodes can expose actions additional actions that can then be called back.
|
||
*/
|
||
class FastActionsButton extends BaseAnyInputConnectedNode {
|
||
static override type = NodeTypesString.FAST_ACTIONS_BUTTON;
|
||
static override title = NodeTypesString.FAST_ACTIONS_BUTTON;
|
||
override comfyClass = NodeTypesString.FAST_ACTIONS_BUTTON;
|
||
|
||
readonly logger = rgthree.newLogSession("[FastActionsButton]");
|
||
|
||
static "@buttonText" = {type: "string"};
|
||
static "@shortcutModifier" = {
|
||
type: "combo",
|
||
values: ["ctrl", "alt", "shift"],
|
||
};
|
||
static "@shortcutKey" = {type: "string"};
|
||
|
||
static collapsible = false;
|
||
|
||
override readonly isVirtualNode = true;
|
||
|
||
override serialize_widgets = true;
|
||
|
||
readonly buttonWidget: IButtonWidget;
|
||
|
||
readonly widgetToData = new Map<IWidget, {comfy?: ComfyApp; node?: LGraphNode}>();
|
||
readonly nodeIdtoFunctionCache = new Map<number, string>();
|
||
|
||
readonly keypressBound;
|
||
readonly keyupBound;
|
||
|
||
private executingFromShortcut = false;
|
||
|
||
override properties!: BaseAnyInputConnectedNode["properties"] & {
|
||
buttonText: string;
|
||
shortcutModifier: string;
|
||
shortcutKey: string;
|
||
};
|
||
|
||
constructor(title?: string) {
|
||
super(title);
|
||
this.properties["buttonText"] = "🎬 Action!";
|
||
this.properties["shortcutModifier"] = "alt";
|
||
this.properties["shortcutKey"] = "";
|
||
this.buttonWidget = this.addWidget(
|
||
"button",
|
||
this.properties["buttonText"],
|
||
"",
|
||
() => {
|
||
this.executeConnectedNodes();
|
||
},
|
||
{serialize: false},
|
||
) as IButtonWidget;
|
||
|
||
this.keypressBound = this.onKeypress.bind(this);
|
||
this.keyupBound = this.onKeyup.bind(this);
|
||
this.onConstructed();
|
||
}
|
||
|
||
/** When we're given data to configure, like from a PNG or JSON. */
|
||
override configure(info: ISerialisedNode): void {
|
||
super.configure(info);
|
||
// Since we add the widgets dynamically, we need to wait to set their values
|
||
// with a short timeout.
|
||
setTimeout(() => {
|
||
if (info.widgets_values) {
|
||
for (let [index, value] of info.widgets_values.entries()) {
|
||
if (index > 0) {
|
||
if (typeof value === "string" && value.startsWith("comfy_action:")) {
|
||
value = value.replace("comfy_action:", "");
|
||
this.addComfyActionWidget(index, value);
|
||
}
|
||
if (this.widgets[index]) {
|
||
this.widgets[index]!.value = value;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}, 100);
|
||
}
|
||
|
||
override clone() {
|
||
const cloned = super.clone()!;
|
||
cloned.properties["buttonText"] = "🎬 Action!";
|
||
cloned.properties["shortcutKey"] = "";
|
||
return cloned;
|
||
}
|
||
|
||
override onAdded(graph: LGraph): void {
|
||
window.addEventListener("keydown", this.keypressBound);
|
||
window.addEventListener("keyup", this.keyupBound);
|
||
}
|
||
|
||
override onRemoved(): void {
|
||
window.removeEventListener("keydown", this.keypressBound);
|
||
window.removeEventListener("keyup", this.keyupBound);
|
||
}
|
||
|
||
async onKeypress(event: KeyboardEvent) {
|
||
const target = (event.target as HTMLElement)!;
|
||
if (
|
||
this.executingFromShortcut ||
|
||
target.localName == "input" ||
|
||
target.localName == "textarea"
|
||
) {
|
||
return;
|
||
}
|
||
if (
|
||
this.properties["shortcutKey"].trim() &&
|
||
this.properties["shortcutKey"].toLowerCase() === event.key.toLowerCase()
|
||
) {
|
||
const shortcutModifier = this.properties["shortcutModifier"];
|
||
let good = shortcutModifier === "ctrl" && event.ctrlKey;
|
||
good = good || (shortcutModifier === "alt" && event.altKey);
|
||
good = good || (shortcutModifier === "shift" && event.shiftKey);
|
||
good = good || (shortcutModifier === "meta" && event.metaKey);
|
||
if (good) {
|
||
setTimeout(() => {
|
||
this.executeConnectedNodes();
|
||
}, 20);
|
||
this.executingFromShortcut = true;
|
||
event.preventDefault();
|
||
event.stopImmediatePropagation();
|
||
app.canvas.dirty_canvas = true;
|
||
return false;
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
|
||
onKeyup(event: KeyboardEvent) {
|
||
const target = (event.target as HTMLElement)!;
|
||
if (target.localName == "input" || target.localName == "textarea") {
|
||
return;
|
||
}
|
||
this.executingFromShortcut = false;
|
||
}
|
||
|
||
override onPropertyChanged(property: string, value: unknown, prevValue?: unknown) {
|
||
if (property == "buttonText" && typeof value === "string") {
|
||
this.buttonWidget.name = value;
|
||
}
|
||
if (property == "shortcutKey" && typeof value === "string") {
|
||
this.properties["shortcutKey"] = value.trim()[0]?.toLowerCase() ?? "";
|
||
}
|
||
return true;
|
||
}
|
||
|
||
override handleLinkedNodesStabilization(linkedNodes: LGraphNode[]) {
|
||
let changed = false;
|
||
// Remove any widgets and data for widgets that are no longer linked.
|
||
for (const [widget, data] of this.widgetToData.entries()) {
|
||
if (!data.node) {
|
||
continue;
|
||
}
|
||
if (!linkedNodes.includes(data.node)) {
|
||
const index = this.widgets.indexOf(widget);
|
||
if (index > -1) {
|
||
this.widgetToData.delete(widget);
|
||
this.removeWidget(widget);
|
||
changed = true;
|
||
} else {
|
||
const [m, a] = this.logger.debugParts("Connected widget is not in widgets... weird.");
|
||
console[m]?.(...a);
|
||
}
|
||
}
|
||
}
|
||
|
||
const badNodes: LGraphNode[] = []; // Nodes that are deleted elsewhere may not exist in linkedNodes.
|
||
let indexOffset = 1; // Start with button, increment when we hit a non-node widget (like comfy)
|
||
for (const [index, node] of linkedNodes.entries()) {
|
||
// Sometimes linkedNodes is stale.
|
||
if (!node) {
|
||
const [m, a] = this.logger.debugParts("linkedNode provided that does not exist. ");
|
||
console[m]?.(...a);
|
||
badNodes.push(node);
|
||
continue;
|
||
}
|
||
let widgetAtSlot = this.widgets[index + indexOffset];
|
||
if (widgetAtSlot && this.widgetToData.get(widgetAtSlot)?.comfy) {
|
||
indexOffset++;
|
||
widgetAtSlot = this.widgets[index + indexOffset];
|
||
}
|
||
|
||
if (!widgetAtSlot || this.widgetToData.get(widgetAtSlot)?.node?.id !== node.id) {
|
||
// Find the next widget that matches the node.
|
||
let widget: IWidget | null = null;
|
||
for (let i = index + indexOffset; i < this.widgets.length; i++) {
|
||
if (this.widgetToData.get(this.widgets[i]!)?.node?.id === node.id) {
|
||
widget = this.widgets.splice(i, 1)[0]!;
|
||
this.widgets.splice(index + indexOffset, 0, widget);
|
||
changed = true;
|
||
break;
|
||
}
|
||
}
|
||
if (!widget) {
|
||
// Add a widget at this spot.
|
||
const exposedActions: string[] = (node.constructor as any).exposedActions || [];
|
||
widget = this.addWidget("combo", node.title, "None", "", {
|
||
values: ["None", "Mute", "Bypass", "Enable", ...exposedActions],
|
||
}) as IWidget;
|
||
widget.serializeValue = async (_node: LGraphNode, _index: number) => {
|
||
return widget?.value;
|
||
};
|
||
this.widgetToData.set(widget, {node});
|
||
changed = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Go backwards through widgets, and remove any that are not in out widgetToData
|
||
for (let i = this.widgets.length - 1; i > linkedNodes.length + indexOffset - 1; i--) {
|
||
const widgetAtSlot = this.widgets[i];
|
||
if (widgetAtSlot && this.widgetToData.get(widgetAtSlot)?.comfy) {
|
||
continue;
|
||
}
|
||
this.removeWidget(widgetAtSlot);
|
||
changed = true;
|
||
}
|
||
return changed;
|
||
}
|
||
|
||
override removeWidget(widget: IBaseWidget | IWidget | number | undefined): void {
|
||
widget = typeof widget === "number" ? this.widgets[widget] : widget;
|
||
if (widget && this.widgetToData.has(widget as IWidget)) {
|
||
this.widgetToData.delete(widget as IWidget);
|
||
}
|
||
super.removeWidget(widget);
|
||
}
|
||
|
||
/**
|
||
* Runs through the widgets, and executes the actions.
|
||
*/
|
||
async executeConnectedNodes() {
|
||
for (const widget of this.widgets) {
|
||
if (widget == this.buttonWidget) {
|
||
continue;
|
||
}
|
||
const action = widget.value;
|
||
const {comfy, node} = this.widgetToData.get(widget) ?? {};
|
||
if (comfy) {
|
||
if (action === "Queue Prompt") {
|
||
await comfy.queuePrompt(0);
|
||
}
|
||
continue;
|
||
}
|
||
if (node) {
|
||
if (action === "Mute") {
|
||
changeModeOfNodes(node, MODE_MUTE);
|
||
} else if (action === "Bypass") {
|
||
changeModeOfNodes(node, MODE_BYPASS);
|
||
} else if (action === "Enable") {
|
||
changeModeOfNodes(node, MODE_ALWAYS);
|
||
}
|
||
// If there's a handleAction, always call it.
|
||
if ((node as RgthreeBaseVirtualNode).handleAction) {
|
||
if (typeof action !== "string") {
|
||
throw new Error("Fast Actions Button action should be a string: " + action);
|
||
}
|
||
await (node as RgthreeBaseVirtualNode).handleAction(action);
|
||
}
|
||
this.graph?.change();
|
||
continue;
|
||
}
|
||
console.warn("Fast Actions Button has a widget without correct data.");
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Adds a ComfyActionWidget at the provided slot (or end).
|
||
*/
|
||
addComfyActionWidget(slot?: number, value?: string) {
|
||
let widget = this.addWidget(
|
||
"combo",
|
||
"Comfy Action",
|
||
"None",
|
||
() => {
|
||
if (String(widget.value).startsWith("MOVE ")) {
|
||
this.widgets.push(this.widgets.splice(this.widgets.indexOf(widget), 1)[0]!);
|
||
widget.value = String(widget.rgthree_lastValue);
|
||
} else if (String(widget.value).startsWith("REMOVE ")) {
|
||
this.removeWidget(widget);
|
||
}
|
||
widget.rgthree_lastValue = widget.value;
|
||
},
|
||
{
|
||
values: ["None", "Queue Prompt", "REMOVE Comfy Action", "MOVE to end"],
|
||
},
|
||
) as IComboWidget;
|
||
widget.rgthree_lastValue = value;
|
||
|
||
widget.serializeValue = async (_node: LGraphNode, _index: number) => {
|
||
return `comfy_app:${widget?.value}`;
|
||
};
|
||
this.widgetToData.set(widget, {comfy: app});
|
||
|
||
if (slot != null) {
|
||
this.widgets.splice(slot, 0, this.widgets.splice(this.widgets.indexOf(widget), 1)[0]!);
|
||
}
|
||
return widget;
|
||
}
|
||
|
||
override onSerialize(serialised: ISerialisedNode) {
|
||
super.onSerialize?.(serialised);
|
||
for (let [index, value] of (serialised.widgets_values || []).entries()) {
|
||
if (this.widgets[index]?.name === "Comfy Action") {
|
||
serialised.widgets_values![index] = `comfy_action:${value}`;
|
||
}
|
||
}
|
||
}
|
||
|
||
static override setUp() {
|
||
super.setUp();
|
||
addMenuItem(this, app, {
|
||
name: "➕ Append a Comfy Action",
|
||
callback: (nodeArg: LGraphNode) => {
|
||
(nodeArg as FastActionsButton).addComfyActionWidget();
|
||
},
|
||
});
|
||
}
|
||
}
|
||
|
||
app.registerExtension({
|
||
name: "rgthree.FastActionsButton",
|
||
registerCustomNodes() {
|
||
FastActionsButton.setUp();
|
||
},
|
||
loadedGraphNode(node: LGraphNode) {
|
||
if (node.type == FastActionsButton.title) {
|
||
(node as FastActionsButton)._tempWidth = node.size[0];
|
||
}
|
||
},
|
||
});
|