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>
99 lines
2.8 KiB
TypeScript
99 lines
2.8 KiB
TypeScript
import type {
|
|
LGraphCanvas as TLGraphCanvas,
|
|
LGraphNode,
|
|
ContextMenu,
|
|
IContextMenuValue,
|
|
IBaseWidget,
|
|
IContextMenuOptions,
|
|
} from "@comfyorg/frontend";
|
|
|
|
import {app} from "scripts/app.js";
|
|
import {rgthreeApi} from "rgthree/common/rgthree_api.js";
|
|
|
|
const PASS_THROUGH = function <T extends any, I extends any>(item: T) {
|
|
return item as T;
|
|
};
|
|
|
|
/**
|
|
* Shows a lora chooser context menu.
|
|
*/
|
|
export async function showLoraChooser(
|
|
event: PointerEvent | MouseEvent,
|
|
callback: IContextMenuOptions["callback"],
|
|
parentMenu?: ContextMenu | null,
|
|
loras?: string[],
|
|
) {
|
|
const canvas = app.canvas as TLGraphCanvas;
|
|
if (!loras) {
|
|
loras = ["None", ...(await rgthreeApi.getLoras().then((loras) => loras.map((l) => l.file)))];
|
|
}
|
|
new LiteGraph.ContextMenu(loras, {
|
|
event: event,
|
|
parentMenu: parentMenu != null ? parentMenu : undefined,
|
|
title: "Choose a lora",
|
|
scale: Math.max(1, canvas.ds?.scale ?? 1),
|
|
className: "dark",
|
|
callback,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Shows a context menu chooser of nodes.
|
|
*
|
|
* @param mapFn The function used to map each node to the context menu item. If null is returned
|
|
* it will be filtered out (rather than use a separate filter method).
|
|
*/
|
|
export function showNodesChooser<T extends IContextMenuValue>(
|
|
event: PointerEvent | MouseEvent,
|
|
mapFn: (n: LGraphNode) => T | null,
|
|
callback: IContextMenuOptions["callback"],
|
|
parentMenu?: ContextMenu,
|
|
) {
|
|
const canvas = app.canvas as TLGraphCanvas;
|
|
const nodesOptions: T[] = (app.graph._nodes as LGraphNode[])
|
|
.map(mapFn)
|
|
.filter((e): e is NonNullable<any> => e != null);
|
|
|
|
nodesOptions.sort((a: any, b: any) => {
|
|
return a.value - b.value;
|
|
});
|
|
|
|
new LiteGraph.ContextMenu(nodesOptions, {
|
|
event: event,
|
|
parentMenu,
|
|
title: "Choose a node id",
|
|
scale: Math.max(1, canvas.ds?.scale ?? 1),
|
|
className: "dark",
|
|
callback,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Shows a conmtext menu chooser for a specific node.
|
|
*
|
|
* @param mapFn The function used to map each node to the context menu item. If null is returned
|
|
* it will be filtered out (rather than use a separate filter method).
|
|
*/
|
|
export function showWidgetsChooser<T extends IContextMenuValue>(
|
|
event: PointerEvent | MouseEvent,
|
|
node: LGraphNode,
|
|
mapFn: (n: IBaseWidget) => T | null,
|
|
callback: IContextMenuOptions["callback"],
|
|
parentMenu?: ContextMenu,
|
|
) {
|
|
const options: T[] = (node.widgets || [])
|
|
.map(mapFn)
|
|
.filter((e): e is NonNullable<any> => e != null);
|
|
if (options.length) {
|
|
const canvas = app.canvas as TLGraphCanvas;
|
|
new LiteGraph.ContextMenu(options, {
|
|
event,
|
|
parentMenu,
|
|
title: "Choose an input/widget",
|
|
scale: Math.max(1, canvas.ds?.scale ?? 1),
|
|
className: "dark",
|
|
callback,
|
|
});
|
|
}
|
|
}
|