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>
309 lines
10 KiB
TypeScript
309 lines
10 KiB
TypeScript
import type {
|
|
IContextMenuOptions,
|
|
ContextMenu,
|
|
LGraphNode as TLGraphNode,
|
|
IWidget,
|
|
LGraphCanvas,
|
|
IContextMenuValue,
|
|
LGraphNodeConstructor,
|
|
ISerialisedNode,
|
|
IButtonWidget,
|
|
} from "@comfyorg/frontend";
|
|
import type {ComfyNodeDef, ComfyApiPrompt} from "typings/comfy.js";
|
|
|
|
import {app} from "scripts/app.js";
|
|
import {ComfyWidgets} from "scripts/widgets.js";
|
|
import {RgthreeBaseServerNode} from "./base_node.js";
|
|
import {rgthree} from "./rgthree.js";
|
|
import {addConnectionLayoutSupport} from "./utils.js";
|
|
import {NodeTypesString} from "./constants.js";
|
|
|
|
const LAST_SEED_BUTTON_LABEL = "♻️ (Use Last Queued Seed)";
|
|
|
|
const SPECIAL_SEED_RANDOM = -1;
|
|
const SPECIAL_SEED_INCREMENT = -2;
|
|
const SPECIAL_SEED_DECREMENT = -3;
|
|
const SPECIAL_SEEDS = [SPECIAL_SEED_RANDOM, SPECIAL_SEED_INCREMENT, SPECIAL_SEED_DECREMENT];
|
|
|
|
interface SeedSerializedCtx {
|
|
inputSeed?: number;
|
|
seedUsed?: number;
|
|
}
|
|
|
|
class RgthreeSeed extends RgthreeBaseServerNode {
|
|
static override title = NodeTypesString.SEED;
|
|
static override type = NodeTypesString.SEED;
|
|
static comfyClass = NodeTypesString.SEED;
|
|
|
|
override serialize_widgets = true;
|
|
|
|
private logger = rgthree.newLogSession(`[Seed]`);
|
|
|
|
static override exposedActions = ["Randomize Each Time", "Use Last Queued Seed"];
|
|
|
|
static "@randomMax" = {type: "number"};
|
|
static "@randomMin" = {type: "number"};
|
|
|
|
lastSeed?: number = undefined;
|
|
serializedCtx: SeedSerializedCtx = {};
|
|
seedWidget!: IWidget;
|
|
lastSeedButton!: IWidget;
|
|
lastSeedValue: IWidget | null = null;
|
|
|
|
private handleApiHijackingBound = this.handleApiHijacking.bind(this);
|
|
|
|
constructor(title = RgthreeSeed.title) {
|
|
super(title);
|
|
|
|
this.properties["randomMax"] = 1125899906842624;
|
|
// We can have a full range of seeds, including negative. But, for the randomRange we'll
|
|
// only generate positives, since that's what folks assume.
|
|
this.properties["randomMin"] = 0;
|
|
|
|
rgthree.addEventListener(
|
|
"comfy-api-queue-prompt-before",
|
|
this.handleApiHijackingBound as EventListener,
|
|
);
|
|
}
|
|
|
|
override onPropertyChanged(prop: string, value: unknown, prevValue?: unknown): boolean {
|
|
if (prop === 'randomMax') {
|
|
this.properties["randomMax"] = Math.min(1125899906842624, Number(value as number));
|
|
} else if (prop === 'randomMin') {
|
|
this.properties["randomMin"] = Math.max(-1125899906842624, Number(value as number));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
override onRemoved() {
|
|
rgthree.addEventListener(
|
|
"comfy-api-queue-prompt-before",
|
|
this.handleApiHijackingBound as EventListener,
|
|
);
|
|
}
|
|
|
|
override configure(info: ISerialisedNode): void {
|
|
super.configure(info);
|
|
if (this.properties?.["showLastSeed"]) {
|
|
this.addLastSeedValue();
|
|
}
|
|
}
|
|
|
|
override async handleAction(action: string) {
|
|
if (action === "Randomize Each Time") {
|
|
this.seedWidget.value = SPECIAL_SEED_RANDOM;
|
|
} else if (action === "Use Last Queued Seed") {
|
|
this.seedWidget.value = this.lastSeed != null ? this.lastSeed : this.seedWidget.value;
|
|
this.lastSeedButton.name = LAST_SEED_BUTTON_LABEL;
|
|
this.lastSeedButton.disabled = true;
|
|
}
|
|
}
|
|
|
|
override onNodeCreated() {
|
|
super.onNodeCreated?.();
|
|
// Grab the already available widgets, and remove the built-in control_after_generate
|
|
for (const [i, w] of this.widgets.entries()) {
|
|
if (w.name === "seed") {
|
|
this.seedWidget = w; // as ComfyWidget;
|
|
this.seedWidget.value = SPECIAL_SEED_RANDOM;
|
|
} else if (w.name === "control_after_generate") {
|
|
this.widgets.splice(i, 1);
|
|
}
|
|
}
|
|
|
|
this.addWidget(
|
|
"button",
|
|
"🎲 Randomize Each Time",
|
|
"",
|
|
() => {
|
|
this.seedWidget.value = SPECIAL_SEED_RANDOM;
|
|
},
|
|
{serialize: false},
|
|
);
|
|
|
|
this.addWidget(
|
|
"button",
|
|
"🎲 New Fixed Random",
|
|
"",
|
|
() => {
|
|
this.seedWidget.value = this.generateRandomSeed();
|
|
},
|
|
{serialize: false},
|
|
);
|
|
|
|
this.lastSeedButton = this.addWidget(
|
|
"button",
|
|
LAST_SEED_BUTTON_LABEL,
|
|
"",
|
|
() => {
|
|
this.seedWidget.value = this.lastSeed != null ? this.lastSeed : this.seedWidget.value;
|
|
this.lastSeedButton.name = LAST_SEED_BUTTON_LABEL;
|
|
this.lastSeedButton.disabled = true;
|
|
},
|
|
{width: 50, serialize: false} as any,
|
|
) as IButtonWidget;
|
|
this.lastSeedButton.disabled = true;
|
|
}
|
|
|
|
generateRandomSeed() {
|
|
let step = this.seedWidget.options.step || 1;
|
|
const randomMin = Number(this.properties['randomMin'] || 0);
|
|
const randomMax = Number(this.properties['randomMax'] || 1125899906842624);
|
|
const randomRange = (randomMax - randomMin) / (step / 10);
|
|
let seed = Math.floor(Math.random() * randomRange) * (step / 10) + randomMin;
|
|
if (SPECIAL_SEEDS.includes(seed)) {
|
|
seed = 0;
|
|
}
|
|
return seed;
|
|
}
|
|
|
|
override getExtraMenuOptions(canvas: LGraphCanvas, options: IContextMenuValue[]) {
|
|
super.getExtraMenuOptions?.apply(this, [...arguments] as any);
|
|
options.splice(options.length - 1, 0, {
|
|
content: "Show/Hide Last Seed Value",
|
|
callback: (
|
|
_value: IContextMenuValue,
|
|
_options: IContextMenuOptions,
|
|
_event: MouseEvent,
|
|
_parentMenu: ContextMenu | undefined,
|
|
_node: TLGraphNode,
|
|
) => {
|
|
this.properties["showLastSeed"] = !this.properties["showLastSeed"];
|
|
if (this.properties["showLastSeed"]) {
|
|
this.addLastSeedValue();
|
|
} else {
|
|
this.removeLastSeedValue();
|
|
}
|
|
},
|
|
});
|
|
return [];
|
|
}
|
|
|
|
addLastSeedValue() {
|
|
if (this.lastSeedValue) return;
|
|
this.lastSeedValue = ComfyWidgets["STRING"](
|
|
this,
|
|
"last_seed",
|
|
["STRING", {multiline: true}],
|
|
app,
|
|
).widget as unknown as IWidget;
|
|
this.lastSeedValue!.inputEl!.readOnly = true;
|
|
this.lastSeedValue!.inputEl!.style.fontSize = "0.75rem";
|
|
this.lastSeedValue!.inputEl!.style.textAlign = "center";
|
|
this.computeSize();
|
|
}
|
|
|
|
removeLastSeedValue() {
|
|
if (!this.lastSeedValue) return;
|
|
this.lastSeedValue!.inputEl!.remove();
|
|
this.widgets.splice(this.widgets.indexOf(this.lastSeedValue), 1);
|
|
this.lastSeedValue = null;
|
|
this.computeSize();
|
|
}
|
|
|
|
/**
|
|
* Intercepts the prompt right before ComfyUI sends it to the server (as fired from rgthree) so we
|
|
* can inspect the prompt and workflow data and change swap in the seeds.
|
|
*
|
|
* Note, the original implementation tried to change the widget value itself when the graph was
|
|
* queued (and the relied on ComfyUI serializing the data changed data) and then changing it back.
|
|
* This worked well until other extensions kept calling graphToPrompt during asynchronous
|
|
* operations within, causing the widget to get confused without a reliable state to reflect upon.
|
|
*/
|
|
handleApiHijacking(e: CustomEvent<ComfyApiPrompt>) {
|
|
// Don't do any work if we're muted/bypassed.
|
|
if (this.mode === LiteGraph.NEVER || this.mode === 4) {
|
|
return;
|
|
}
|
|
|
|
const workflow = e.detail.workflow;
|
|
const output = e.detail.output;
|
|
|
|
let workflowNode = workflow?.nodes?.find((n: ISerialisedNode) => n.id === this.id) ?? null;
|
|
let outputInputs = output?.[this.id]?.inputs;
|
|
|
|
if (
|
|
!workflowNode ||
|
|
!outputInputs ||
|
|
outputInputs[this.seedWidget.name || "seed"] === undefined
|
|
) {
|
|
const [n, v] = this.logger.warnParts(
|
|
`Node ${this.id} not found in prompt data sent to server. This may be fine if only ` +
|
|
`queuing part of the workflow. If not, then this could be a bug.`,
|
|
);
|
|
console[n]?.(...v);
|
|
return;
|
|
}
|
|
|
|
const seedToUse = this.getSeedToUse();
|
|
const seedWidgetndex = this.widgets.indexOf(this.seedWidget);
|
|
|
|
workflowNode.widgets_values![seedWidgetndex] = seedToUse;
|
|
outputInputs[this.seedWidget.name || "seed"] = seedToUse;
|
|
|
|
this.lastSeed = seedToUse;
|
|
if (seedToUse != this.seedWidget.value) {
|
|
this.lastSeedButton.name = `♻️ ${this.lastSeed}`;
|
|
this.lastSeedButton.disabled = false;
|
|
} else {
|
|
this.lastSeedButton.name = LAST_SEED_BUTTON_LABEL;
|
|
this.lastSeedButton.disabled = true;
|
|
}
|
|
if (this.lastSeedValue) {
|
|
this.lastSeedValue.value = `Last Seed: ${this.lastSeed}`;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Determines a seed to use depending on the seed widget's current value and the last used seed.
|
|
* There are no sideffects to calling this method.
|
|
*/
|
|
private getSeedToUse() {
|
|
const inputSeed = Number(this.seedWidget.value);
|
|
let seedToUse: number | null = null;
|
|
|
|
// If our input seed was a special seed, then handle it.
|
|
if (SPECIAL_SEEDS.includes(inputSeed)) {
|
|
// If the last seed was not a special seed and we have increment/decrement, then do that on
|
|
// the last seed.
|
|
if (typeof this.lastSeed === "number" && !SPECIAL_SEEDS.includes(this.lastSeed)) {
|
|
if (inputSeed === SPECIAL_SEED_INCREMENT) {
|
|
seedToUse = this.lastSeed + 1;
|
|
} else if (inputSeed === SPECIAL_SEED_DECREMENT) {
|
|
seedToUse = this.lastSeed - 1;
|
|
}
|
|
}
|
|
// If we don't have a seed to use, or it's special seed (like we incremented into one), then
|
|
// we randomize.
|
|
if (seedToUse == null || SPECIAL_SEEDS.includes(seedToUse)) {
|
|
seedToUse = this.generateRandomSeed();
|
|
}
|
|
}
|
|
|
|
return seedToUse ?? inputSeed;
|
|
}
|
|
|
|
static override setUp(comfyClass: typeof LGraphNode, nodeData: ComfyNodeDef) {
|
|
RgthreeBaseServerNode.registerForOverride(comfyClass, nodeData, RgthreeSeed);
|
|
}
|
|
|
|
static override onRegisteredForOverride(comfyClass: any, ctxClass: any) {
|
|
addConnectionLayoutSupport(RgthreeSeed, app, [
|
|
["Left", "Right"],
|
|
["Right", "Left"],
|
|
]);
|
|
setTimeout(() => {
|
|
RgthreeSeed.category = comfyClass.category;
|
|
});
|
|
}
|
|
}
|
|
|
|
app.registerExtension({
|
|
name: "rgthree.Seed",
|
|
async beforeRegisterNodeDef(nodeType: typeof LGraphNode, nodeData: ComfyNodeDef) {
|
|
if (nodeData.name === RgthreeSeed.type) {
|
|
RgthreeSeed.setUp(nodeType, nodeData);
|
|
}
|
|
},
|
|
});
|