diff --git a/src/lib/widgets/ImageEditorWidget.svelte b/ImageUploadWidget.svelte similarity index 100% rename from src/lib/widgets/ImageEditorWidget.svelte rename to ImageUploadWidget.svelte diff --git a/src/lib/ComfyGraph.ts b/src/lib/ComfyGraph.ts index 6eb2561..d1cdd91 100644 --- a/src/lib/ComfyGraph.ts +++ b/src/lib/ComfyGraph.ts @@ -8,7 +8,7 @@ import { get } from "svelte/store"; import type ComfyGraphNode from "./nodes/ComfyGraphNode"; import type IComfyInputSlot from "./IComfyInputSlot"; import type { ComfyBackendNode } from "./nodes/ComfyBackendNode"; -import type { ComfyComboNode, ComfyWidgetNode } from "./nodes"; +import type { ComfyComboNode, ComfyWidgetNode } from "./nodes/widgets"; type ComfyGraphEvents = { configured: (graph: LGraph) => void diff --git a/src/lib/api.ts b/src/lib/api.ts index 14e860b..445a01e 100644 --- a/src/lib/api.ts +++ b/src/lib/api.ts @@ -1,7 +1,7 @@ import type { Progress, SerializedPrompt, SerializedPromptInputs, SerializedPromptInputsAll, SerializedPromptOutputs } from "./components/ComfyApp"; import type TypedEmitter from "typed-emitter"; import EventEmitter from "events"; -import type { ComfyExecutionResult, ComfyImageLocation } from "./nodes/ComfyWidgetNodes"; +import type { ComfyImageLocation } from "$lib/utils"; import type { SerializedLGraph, UUID } from "@litegraph-ts/core"; import type { SerializedLayoutState } from "./stores/layoutState"; import type { ComfyNodeDef } from "./ComfyNodeDef"; diff --git a/src/lib/components/BlockContainer.svelte b/src/lib/components/BlockContainer.svelte index f7a8c57..dd062f8 100644 --- a/src/lib/components/BlockContainer.svelte +++ b/src/lib/components/BlockContainer.svelte @@ -23,13 +23,21 @@ export let dragDisabled: boolean = false; export let isMobile: boolean = false; - let attrsChanged: Writable | null = null; + let attrsChanged: Writable | null = null; let children: IDragItem[] | null = null; const flipDurationMs = 100; $: if (container) { - children = $layoutState.allItems[container.id].children; - attrsChanged = container.attrsChanged + const entry = $layoutState.allItems[container.id] + if (entry) { + children = $layoutState.allItems[container.id].children; + attrsChanged = container.attrsChanged + } + else { + container = null; + children = null; + attrsChanged = null; + } } else { children = null; diff --git a/src/lib/components/ComfyApp.ts b/src/lib/components/ComfyApp.ts index 6d243fa..72636a7 100644 --- a/src/lib/components/ComfyApp.ts +++ b/src/lib/components/ComfyApp.ts @@ -12,11 +12,12 @@ import "@litegraph-ts/nodes-logic" import "@litegraph-ts/nodes-math" import "@litegraph-ts/nodes-strings" import "$lib/nodes/index" +import "$lib/nodes/widgets/index" import * as nodes from "$lib/nodes/index" +import * as widgets from "$lib/nodes/widgets/index" import ComfyGraphCanvas, { type SerializedGraphCanvasState } from "$lib/ComfyGraphCanvas"; import type ComfyGraphNode from "$lib/nodes/ComfyGraphNode"; -import * as widgets from "$lib/widgets/index" import queueState from "$lib/stores/queueState"; import { type SvelteComponentDev } from "svelte/internal"; import type IComfyInputSlot from "$lib/IComfyInputSlot"; @@ -32,9 +33,10 @@ import { download, graphToGraphVis, jsonToJsObject, promptToGraphVis, range, wor import notify from "$lib/notify"; import configState from "$lib/stores/configState"; import { blankGraph } from "$lib/defaultGraph"; -import type { ComfyExecutionResult } from "$lib/nodes/ComfyWidgetNodes"; +import type { ComfyExecutionResult } from "$lib/utils"; import ComfyPromptSerializer from "./ComfyPromptSerializer"; import { iterateNodeDefInputs, type ComfyNodeDef, isBackendNodeDefInputType, iterateNodeDefOutputs } from "$lib/ComfyNodeDef"; +import { ComfyComboNode } from "$lib/nodes/widgets"; export const COMFYBOX_SERIAL_VERSION = 1; @@ -77,7 +79,7 @@ export type Progress = { } type BackendComboNode = { - comboNode: nodes.ComfyComboNode + comboNode: ComfyComboNode, inputSlot: IComfyInputSlot, backendNode: ComfyBackendNode } @@ -747,7 +749,7 @@ export default class ComfyApp { }); for (const inputIndex of found) { - const comboNode = backendNode.getInputNode(inputIndex) as nodes.ComfyComboNode + const comboNode = backendNode.getInputNode(inputIndex) as ComfyComboNode const inputSlot = backendNode.inputs[inputIndex] as IComfyInputSlot; const def = defs[backendNode.type]; @@ -762,14 +764,14 @@ export default class ComfyApp { console.debug("[refreshComboInNodes] found:", backendUpdatedCombos.length, backendUpdatedCombos) // Mark combo nodes without backend configs as being loaded already. - for (const node of this.lGraph.iterateNodesOfClassRecursive(nodes.ComfyComboNode)) { + for (const node of this.lGraph.iterateNodesOfClassRecursive(ComfyComboNode)) { if (backendUpdatedCombos[node.id] != null) { continue; } // This node isn't connected to a backend node, so it's configured // by the frontend instead. - const comboNode = node as nodes.ComfyComboNode; + const comboNode = node as ComfyComboNode; let values = comboNode.properties.values; // Frontend nodes can declare defaultWidgets which creates a diff --git a/src/lib/components/WidgetContainer.svelte b/src/lib/components/WidgetContainer.svelte index 9c5454c..bc9bbb7 100644 --- a/src/lib/components/WidgetContainer.svelte +++ b/src/lib/components/WidgetContainer.svelte @@ -7,8 +7,7 @@ import { startDrag, stopDrag } from "$lib/utils" import Container from "./Container.svelte" import { type Writable } from "svelte/store" - import type { ComfyWidgetNode } from "$lib/nodes"; - import { NodeMode } from "@litegraph-ts/core"; + import type { ComfyWidgetNode } from "$lib/nodes/widgets"; import { isHidden } from "$lib/widgets/utils"; export let dragItem: IDragItem | null = null; diff --git a/src/lib/components/gradio/app/Row.svelte b/src/lib/components/gradio/app/Row.svelte index 84a61fa..c009673 100644 --- a/src/lib/components/gradio/app/Row.svelte +++ b/src/lib/components/gradio/app/Row.svelte @@ -2,7 +2,7 @@ import type { Styles } from "@gradio/utils"; export let style: Styles = {}; - export let elem_id: string; + export let elem_id: string | null; export let elem_classes: Array = []; export let visible: boolean = true; export let variant: "default" | "panel" | "compact" = "default"; diff --git a/src/lib/nodes/ComfyActionNodes.ts b/src/lib/nodes/ComfyActionNodes.ts index df614af..1aad2eb 100644 --- a/src/lib/nodes/ComfyActionNodes.ts +++ b/src/lib/nodes/ComfyActionNodes.ts @@ -5,10 +5,10 @@ import queueState from "$lib/stores/queueState"; import { BuiltInSlotType, LiteGraph, NodeMode, type ITextWidget, type IToggleWidget, type SerializedLGraphNode, type SlotLayout, type PropertyLayout } from "@litegraph-ts/core"; import { get } from "svelte/store"; import ComfyGraphNode, { type ComfyGraphNodeProperties } from "./ComfyGraphNode"; -import type { ComfyWidgetNode, ComfyExecutionResult, ComfyImageLocation } from "./ComfyWidgetNodes"; +import type { ComfyWidgetNode } from "$lib/nodes/widgets"; import type { NotifyOptions } from "$lib/notify"; import type { FileData as GradioFileData } from "@gradio/upload"; -import { convertComfyOutputToGradio, type ComfyUploadImageAPIResponse } from "$lib/utils"; +import { type ComfyExecutionResult, type ComfyImageLocation, convertComfyOutputToGradio, type ComfyUploadImageAPIResponse } from "$lib/utils"; export class ComfyQueueEvents extends ComfyGraphNode { static slotLayout: SlotLayout = { diff --git a/src/lib/nodes/ComfyBackendNode.ts b/src/lib/nodes/ComfyBackendNode.ts index 1bf0f5c..c899b19 100644 --- a/src/lib/nodes/ComfyBackendNode.ts +++ b/src/lib/nodes/ComfyBackendNode.ts @@ -1,11 +1,12 @@ import LGraphCanvas from "@litegraph-ts/core/src/LGraphCanvas"; import ComfyGraphNode from "./ComfyGraphNode"; import ComfyWidgets from "$lib/widgets" -import type { ComfyWidgetNode, ComfyExecutionResult } from "./ComfyWidgetNodes"; +import type { ComfyWidgetNode } from "$lib/nodes/widgets"; import { BuiltInSlotShape, BuiltInSlotType, type SerializedLGraphNode } from "@litegraph-ts/core"; import type IComfyInputSlot from "$lib/IComfyInputSlot"; import type { ComfyInputConfig } from "$lib/IComfyInputSlot"; import { iterateNodeDefOutputs, type ComfyNodeDef, iterateNodeDefInputs } from "$lib/ComfyNodeDef"; +import type { ComfyExecutionResult } from "$lib/utils"; /* * Base class for any node with configuration sent by the backend. diff --git a/src/lib/nodes/ComfyGraphNode.ts b/src/lib/nodes/ComfyGraphNode.ts index 367ae19..1fb1883 100644 --- a/src/lib/nodes/ComfyGraphNode.ts +++ b/src/lib/nodes/ComfyGraphNode.ts @@ -1,9 +1,9 @@ import type { ComfyInputConfig } from "$lib/IComfyInputSlot"; import type { SerializedPrompt } from "$lib/components/ComfyApp"; -import type ComfyWidget from "$lib/components/widgets/ComfyWidget"; import { LGraph, LGraphNode, LLink, LiteGraph, NodeMode, type INodeInputSlot, type SerializedLGraphNode, type Vector2, type INodeOutputSlot, LConnectionKind, type SlotType, LGraphCanvas, getStaticPropertyOnInstance, type PropertyLayout, type SlotLayout } from "@litegraph-ts/core"; import type { SvelteComponentDev } from "svelte/internal"; -import type { ComfyWidgetNode, ComfyExecutionResult, ComfyImageLocation } from "./ComfyWidgetNodes"; +import type { ComfyWidgetNode } from "$lib/nodes/widgets"; +import type { ComfyExecutionResult, ComfyImageLocation } from "$lib/utils" import type IComfyInputSlot from "$lib/IComfyInputSlot"; import uiState from "$lib/stores/uiState"; import { get } from "svelte/store"; diff --git a/src/lib/nodes/ComfyValueControl.ts b/src/lib/nodes/ComfyValueControl.ts index 7a3485a..b699ee7 100644 --- a/src/lib/nodes/ComfyValueControl.ts +++ b/src/lib/nodes/ComfyValueControl.ts @@ -2,7 +2,7 @@ import { BuiltInSlotType, LiteGraph, type SlotLayout } from "@litegraph-ts/core" import ComfyGraphNode, { type ComfyGraphNodeProperties, type DefaultWidgetLayout } from "./ComfyGraphNode"; import { clamp } from "$lib/utils"; import ComboWidget from "$lib/widgets/ComboWidget.svelte"; -import { ComfyComboNode } from "./ComfyWidgetNodes"; +import { ComfyComboNode } from "./widgets"; export interface ComfyValueControlProperties extends ComfyGraphNodeProperties { value: any, diff --git a/src/lib/nodes/ComfyWidgetNodes.ts b/src/lib/nodes/ComfyWidgetNodes.ts deleted file mode 100644 index f779b7d..0000000 --- a/src/lib/nodes/ComfyWidgetNodes.ts +++ /dev/null @@ -1,897 +0,0 @@ -import { LiteGraph, type ContextMenuItem, type LGraphNode, type Vector2, LConnectionKind, LLink, LGraphCanvas, type SlotType, TitleMode, type SlotLayout, LGraph, type INodeInputSlot, type ITextWidget, type INodeOutputSlot, type SerializedLGraphNode, BuiltInSlotType, type PropertyLayout, type IComboWidget, NodeMode, type INumberWidget, type UUID } from "@litegraph-ts/core"; -import ComfyGraphNode, { type ComfyGraphNodeProperties } from "./ComfyGraphNode"; -import type { SvelteComponentDev } from "svelte/internal"; -import { Watch } from "@litegraph-ts/nodes-basic"; -import type IComfyInputSlot from "$lib/IComfyInputSlot"; -import { writable, type Unsubscriber, type Writable, get } from "svelte/store"; -import { clamp, convertComfyOutputToGradio, range, type ComfyUploadImageType, isComfyBoxImageMetadata, filenameToComfyBoxMetadata, type ComfyBoxImageMetadata, isComfyExecutionResult, executionResultToImageMetadata, parseWhateverIntoImageMetadata } from "$lib/utils" -import layoutState from "$lib/stores/layoutState"; -import type { FileData as GradioFileData } from "@gradio/upload"; -import queueState from "$lib/stores/queueState"; - -import ComboWidget from "$lib/widgets/ComboWidget.svelte"; -import RangeWidget from "$lib/widgets/RangeWidget.svelte"; -import TextWidget from "$lib/widgets/TextWidget.svelte"; -import GalleryWidget from "$lib/widgets/GalleryWidget.svelte"; -import ButtonWidget from "$lib/widgets/ButtonWidget.svelte"; -import CheckboxWidget from "$lib/widgets/CheckboxWidget.svelte"; -import RadioWidget from "$lib/widgets/RadioWidget.svelte"; -import ImageUploadWidget from "$lib/widgets/ImageUploadWidget.svelte"; -import ImageEditorWidget from "$lib/widgets/ImageEditorWidget.svelte"; -import type { ComfyNodeID } from "$lib/api"; - -export type AutoConfigOptions = { - includeProperties?: Set | null, - setDefaultValue?: boolean - setWidgetTitle?: boolean -} - -/* - * NOTE: If you want to add a new widget but it has the same input/output type - * as another one of the existing widgets, best to create a new "variant" of - * that widget instead. - * - * - Go to layoutState, look for `ALL_ATTRIBUTES,` insert or find a "variant" - * attribute and set `validNodeTypes` to the type of the litegraph node - * - Add a new entry in the `values` array, like "knob" or "dial" for ComfySliderWidget - * - Add an {#if widget.attrs.variant === <...>} statement in the existing Svelte component - * - * Also, BEWARE of calling setOutputData() and triggerSlot() on the same frame! - * You will have to either implement an internal delay on the event triggering - * or use an Event Delay node to ensure the output slot data can propagate to - * the rest of the graph first (see `delayChangedEvent` for details) - */ - -export interface ComfyWidgetProperties extends ComfyGraphNodeProperties { - defaultValue: any -} - -/* - * A node that is tied to a UI widget in the frontend. When the frontend's - * widget is changed, the value of the first output in the node is updated - * in the litegraph instance. - */ -export abstract class ComfyWidgetNode extends ComfyGraphNode { - abstract properties: ComfyWidgetProperties; - - value: Writable - propsChanged: Writable = writable(0) // dummy to indicate if props changed - unsubscribe: Unsubscriber; - - /** Svelte class for the frontend logic */ - abstract svelteComponentType: typeof SvelteComponentDev - - /** If false, user manually set min/max/step, and should not be autoinherited from connected input */ - autoConfig: boolean = true; - - copyFromInputLink: boolean = true; - - /** - * If true wait until next frame update to trigger the changed event. - * Reason is, if the event is triggered immediately then other stuff that wants to run - * their own onExecute on the output value won't have completed yet. - */ - delayChangedEvent: boolean = true; - - private _aboutToChange: number = 0; - private _aboutToChangeValue: any = null; - private _noChangedEvent: boolean = false; - - abstract defaultValue: T; - - /** Names of properties to add as inputs */ - // shownInputProperties: string[] = [] - - /** Names of properties to add as outputs */ - private shownOutputProperties: Record = {} - outputProperties: { name: string, type: string }[] = [] - - override isBackendNode = false; - override serialize_widgets = true; - - storeActionName: string | null = "store"; - - // output slots - outputSlotName: string | null = "value"; - changedEventName: string | null = "changed"; - - displayWidget: ITextWidget; - - override size: Vector2 = [60, 40]; - - constructor(name: string, value: T) { - const color = LGraphCanvas.node_colors["blue"] - super(name) - this.value = writable(value) - this.color ||= color.color - this.bgColor ||= color.bgColor - this.displayWidget = this.addWidget( - "text", - "Value", - "" - ); - this.displayWidget.disabled = true; // prevent editing - this.unsubscribe = this.value.subscribe(this.onValueUpdated.bind(this)) - } - - addPropertyAsOutput(propertyName: string, type: string) { - if (this.shownOutputProperties["@" + propertyName]) - return; - - if (!(propertyName in this.properties)) { - throw `No property named ${propertyName} found!` - } - - this.shownOutputProperties["@" + propertyName] = { type, index: this.outputs.length } - this.addOutput("@" + propertyName, type) - } - - formatValue(value: any): string { - return Watch.toString(value) - } - - override changeMode(modeTo: NodeMode): boolean { - const result = super.changeMode(modeTo); - this.notifyPropsChanged(); - return result; - } - - private onValueUpdated(value: any) { - // console.debug("[Widget] valueUpdated", this, value) - this.displayWidget.value = this.formatValue(value) - - if (this.outputSlotName !== null) { - const outputIndex = this.findOutputSlotIndexByName(this.outputSlotName) - if (outputIndex !== -1) - this.setOutputData(outputIndex, get(this.value)) - } - - if (this.changedEventName !== null && !this._noChangedEvent) { - if (!this.delayChangedEvent) - this.triggerChangeEvent(get(this.value)) - else { - // console.debug("[Widget] queueChangeEvent", this, value) - this._aboutToChange = 2; // wait 1.5-2 frames, in case we're already in the middle of executing the graph - this._aboutToChangeValue = get(this.value); - } - } - this._noChangedEvent = false; - } - - private triggerChangeEvent(value: any) { - // console.debug("[Widget] trigger changed", this, value) - this.trigger(this.changedEventName, value) - } - - parseValue(value: any): T { return value as T }; - - getValue(): T { - return get(this.value); - } - - setValue(value: any, noChangedEvent: boolean = false) { - if (noChangedEvent) - this._noChangedEvent = true; - - const parsed = this.parseValue(value) - this.value.set(parsed) - - // In case value.set() does not trigger onValueUpdated, we need to reset - // the counter here also. - this._noChangedEvent = false; - } - - override onPropertyChanged(property: string, value: any, prevValue?: any) { - if (this.shownOutputProperties != null) { - const data = this.shownOutputProperties[property] - if (data) - this.setOutputData(data.index, value) - } - } - - /* - * Logic to run if this widget can be treated as output (slider, combo, text) - */ - override onExecute(param: any, options: object) { - if (this.outputSlotName != null) { - const outputIndex = this.findOutputSlotIndexByName(this.outputSlotName) - if (outputIndex !== -1) - this.setOutputData(outputIndex, get(this.value)) - } - for (const propName in this.shownOutputProperties) { - const data = this.shownOutputProperties[propName] - this.setOutputData(data.index, this.properties[propName]) - } - - // Fire a pending change event after one full step of the graph has - // finished processing - if (this._aboutToChange > 0) { - this._aboutToChange -= 1 - if (this._aboutToChange <= 0) { - const value = this._aboutToChangeValue; - this._aboutToChange = 0; - this._aboutToChangeValue = null; - this.triggerChangeEvent(value); - } - } - } - - override onAction(action: any, param: any, options: { action_call?: string }) { - if (action === this.storeActionName) { - let noChangedEvent = false; - let value = param; - if (param != null && typeof param === "object" && "value" in param) { - value = param.value - if ("noChangedEvent" in param) - noChangedEvent = Boolean(param.noChangedEvent) - } - this.setValue(value, noChangedEvent) - } - } - - onConnectOutput( - outputIndex: number, - inputType: INodeInputSlot["type"], - input: INodeInputSlot, - inputNode: LGraphNode, - inputIndex: number - ): boolean { - const anyConnected = range(this.outputs.length).some(i => this.getOutputLinks(i).length > 0); - - if (this.autoConfig && "config" in input && !anyConnected && (input as IComfyInputSlot).widgetNodeType === this.type) { - this.doAutoConfig(input as IComfyInputSlot) - } - - return true; - } - - doAutoConfig(input: IComfyInputSlot, options: AutoConfigOptions = { setDefaultValue: true, setWidgetTitle: true }) { - // Copy properties from default config in input slot - const comfyInput = input as IComfyInputSlot; - for (const key in comfyInput.config) { - if (options.includeProperties == null || options.includeProperties.has(key)) - this.setProperty(key, comfyInput.config[key]) - } - - if (options.setDefaultValue) { - if ("defaultValue" in this.properties) - this.setValue(this.properties.defaultValue) - } - - if (options.setWidgetTitle) { - const widget = layoutState.findLayoutForNode(this.id as ComfyNodeID) - if (widget && input.name !== "") { - widget.attrs.title = input.name; - } - } - - // console.debug("Property copy", input, this.properties) - - this.setValue(get(this.value)) - - this.onAutoConfig(input); - - this.notifyPropsChanged(); - } - - onAutoConfig(input: IComfyInputSlot) { - } - - notifyPropsChanged() { - const layoutEntry = layoutState.findLayoutEntryForNode(this.id as ComfyNodeID) - if (layoutEntry && layoutEntry.parent) { - layoutEntry.parent.attrsChanged.set(get(layoutEntry.parent.attrsChanged) + 1) - } - // console.debug("propsChanged", this) - this.propsChanged.set(get(this.propsChanged) + 1) - - } - - override onConnectionsChange( - type: LConnectionKind, - slotIndex: number, - isConnected: boolean, - link: LLink, - ioSlot: (INodeOutputSlot | INodeInputSlot) - ): void { - super.onConnectionsChange(type, slotIndex, isConnected, link, ioSlot); - this.clampConfig(); - } - - clampConfig() { - let changed = false; - for (const link of this.getOutputLinks(0)) { - if (link) { // can be undefined if the link is removed - const node = this.graph._nodes_by_id[link.target_id] - if (node) { - const input = node.inputs[link.target_slot] - if (input && "config" in input) { - this.clampOneConfig(input as IComfyInputSlot) - changed = true; - } - } - } - } - - // Force reactivity change so the frontend can be updated with the new props - this.notifyPropsChanged(); - } - - clampOneConfig(input: IComfyInputSlot) { } - - override onSerialize(o: SerializedLGraphNode) { - (o as any).comfyValue = get(this.value); - (o as any).shownOutputProperties = this.shownOutputProperties - super.onSerialize(o); - } - - override onConfigure(o: SerializedLGraphNode) { - const value = (o as any).comfyValue || LiteGraph.cloneObject(this.defaultValue); - this.value.set(value); - this.shownOutputProperties = (o as any).shownOutputProperties; - } - - override stripUserState(o: SerializedLGraphNode) { - super.stripUserState(o); - (o as any).comfyValue = this.defaultValue; - o.properties.defaultValue = null; - } -} - -export interface ComfySliderProperties extends ComfyWidgetProperties { - min: number, - max: number, - step: number, - precision: number -} - -export class ComfySliderNode extends ComfyWidgetNode { - override properties: ComfySliderProperties = { - tags: [], - defaultValue: 0, - min: 0, - max: 10, - step: 1, - precision: 1 - } - - override svelteComponentType = RangeWidget - override defaultValue = 0; - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "value", type: "number" }, - { name: "changed", type: BuiltInSlotType.EVENT }, - ] - } - - override outputProperties = [ - { name: "min", type: "number" }, - { name: "max", type: "number" }, - { name: "step", type: "number" }, - { name: "precision", type: "number" }, - ] - - constructor(name?: string) { - super(name, 0) - } - - override parseValue(value: any): number { - if (typeof value !== "number") - return this.properties.min; - return clamp(value, this.properties.min, this.properties.max) - } - - override clampOneConfig(input: IComfyInputSlot) { - // this.setProperty("min", clamp(this.properties.min, input.config.min, input.config.max)) - // this.setProperty("max", clamp(this.properties.max, input.config.max, input.config.min)) - // this.setProperty("step", Math.min(this.properties.step, input.config.step)) - this.setValue(this.properties.defaultValue) - } -} - -LiteGraph.registerNodeType({ - class: ComfySliderNode, - title: "UI.Slider", - desc: "Slider outputting a number value", - type: "ui/slider" -}) - -export interface ComfyComboProperties extends ComfyWidgetProperties { - values: string[] - - /* JS Function body that takes a parameter named "value" as a parameter and returns the label for each combo entry */ - convertValueToLabelCode: string -} - -export class ComfyComboNode extends ComfyWidgetNode { - override properties: ComfyComboProperties = { - tags: [], - defaultValue: "A", - values: ["A", "B", "C", "D"], - convertValueToLabelCode: "" - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "value", type: "string" }, - { name: "changed", type: BuiltInSlotType.EVENT } - ] - } - - override svelteComponentType = ComboWidget - override defaultValue = "A"; - override saveUserState = false; - - // True if at least one combo box refresh has taken place - // Wait until the initial graph load for combo to be valid. - firstLoad: Writable; - valuesForCombo: Writable; // Changed when the combo box has values. - - constructor(name?: string) { - super(name, "A") - this.firstLoad = writable(false) - this.valuesForCombo = writable(null) - } - - override onPropertyChanged(property: any, value: any) { - if (property === "values" || property === "convertValueToLabelCode") { - // this.formatValues(this.properties.values) - } - } - - formatValues(values: string[]) { - if (values == null) - return; - - this.properties.values = values; - - let formatter: any; - if (this.properties.convertValueToLabelCode) - formatter = new Function("value", this.properties.convertValueToLabelCode) as (v: string) => string; - else - formatter = (value: any) => `${value}`; - - let valuesForCombo = [] - - try { - valuesForCombo = this.properties.values.map((value, index) => { - return { - value, - label: formatter(value), - index - } - }) - } - catch (err) { - console.error("Failed formatting!", err) - valuesForCombo = this.properties.values.map((value, index) => { - return { - value, - label: `${value}`, - index - } - }) - } - - this.firstLoad.set(true) - this.valuesForCombo.set(valuesForCombo); - } - - onConnectOutput( - outputIndex: number, - inputType: INodeInputSlot["type"], - input: INodeInputSlot, - inputNode: LGraphNode, - inputIndex: number - ): boolean { - if (!super.onConnectOutput(outputIndex, inputType, input, inputNode, inputIndex)) - return false; - - const thisProps = this.properties; - if (!("config" in input)) - return true; - - const comfyInput = input as IComfyInputSlot; - const otherProps = comfyInput.config; - - // Ensure combo options match - if (!(otherProps.values instanceof Array)) - return false; - if (thisProps.values.find((v, i) => otherProps.values.indexOf(v) === -1)) - return false; - - return true; - } - - override parseValue(value: any): string { - if (typeof value !== "string" || this.properties.values.indexOf(value) === -1) - return this.properties.values[0] - return value - } - - override clampOneConfig(input: IComfyInputSlot) { - if (!input.config.values) - this.setValue("") - else if (input.config.values.indexOf(this.properties.value) === -1) { - if (input.config.values.length === 0) - this.setValue("") - else - this.setValue(input.config.defaultValue || input.config.values[0]) - } - } - - override onSerialize(o: SerializedLGraphNode) { - super.onSerialize(o); - // TODO fix saving combo nodes with huge values lists - o.properties.values = [] - } - - override stripUserState(o: SerializedLGraphNode) { - super.stripUserState(o); - o.properties.values = [] - } -} - -LiteGraph.registerNodeType({ - class: ComfyComboNode, - title: "UI.Combo", - desc: "Combo box outputting a string value", - type: "ui/combo" -}) - -export interface ComfyTextProperties extends ComfyWidgetProperties { - multiline: boolean; -} - -export class ComfyTextNode extends ComfyWidgetNode { - override properties: ComfyTextProperties = { - tags: [], - defaultValue: "", - multiline: false - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "value", type: "string" }, - { name: "changed", type: BuiltInSlotType.EVENT } - ] - } - - override svelteComponentType = TextWidget - override defaultValue = ""; - - constructor(name?: string) { - super(name, "") - } - - override parseValue(value: any): string { - return `${value}` - } -} - -LiteGraph.registerNodeType({ - class: ComfyTextNode, - title: "UI.Text", - desc: "Textbox outputting a string value", - type: "ui/text" -}) - -/** Raw output as received from ComfyUI's backend */ -export interface ComfyExecutionResult { - // Technically this response can contain arbitrary data, but "images" is the - // most frequently used as it's output by LoadImage and PreviewImage, the - // only two output nodes in base ComfyUI. - images: ComfyImageLocation[] | null, -} - -/** Raw output entry as received from ComfyUI's backend */ -export type ComfyImageLocation = { - /* Filename with extension in the subfolder. */ - filename: string, - /* Subfolder in the containing folder. */ - subfolder: string, - /* Base ComfyUI folder where the image is located. */ - type: ComfyUploadImageType -} - -export interface ComfyGalleryProperties extends ComfyWidgetProperties { - index: number, - updateMode: "replace" | "append", -} - -export class ComfyGalleryNode extends ComfyWidgetNode { - override properties: ComfyGalleryProperties = { - tags: [], - defaultValue: [], - index: 0, - updateMode: "replace", - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "images", type: "OUTPUT" }, - { name: "store", type: BuiltInSlotType.ACTION, options: { color_off: "rebeccapurple", color_on: "rebeccapurple" } } - ], - outputs: [ - { name: "images", type: "COMFYBOX_IMAGES" }, - { name: "selected_index", type: "number" }, - ] - } - - static propertyLayout: PropertyLayout = [ - { name: "updateMode", defaultValue: "replace", type: "enum", options: { values: ["replace", "append"] } } - ] - - override svelteComponentType = GalleryWidget - override defaultValue = [] - override saveUserState = false; - override outputSlotName = null; - override changedEventName = null; - - selectedFilename: string | null = null; - - selectedIndexWidget: ITextWidget; - modeWidget: IComboWidget; - - constructor(name?: string) { - super(name, []) - this.selectedIndexWidget = this.addWidget("text", "Selected", String(this.properties.index), "index") - this.selectedIndexWidget.disabled = true; - this.modeWidget = this.addWidget("combo", "Mode", this.properties.updateMode, null, { property: "updateMode", values: ["replace", "append"] }) - } - - override onPropertyChanged(property: any, value: any) { - if (property === "updateMode") { - this.modeWidget.value = value; - } - } - - override onExecute() { - this.setOutputData(0, get(this.value)) - this.setOutputData(1, this.properties.index) - } - - override onAction(action: any, param: any, options: { action_call?: string }) { - super.onAction(action, param, options) - } - - override formatValue(value: ComfyBoxImageMetadata[] | null): string { - return `Images: ${value?.length || 0}` - } - - override parseValue(param: any): ComfyBoxImageMetadata[] { - const meta = parseWhateverIntoImageMetadata(param) || []; - - console.debug("[ComfyGalleryNode] Received output!", param) - - if (this.properties.updateMode === "append") { - const currentValue = get(this.value) - return currentValue.concat(meta) - } - else { - return meta; - } - } - - override setValue(value: any, noChangedEvent: boolean = false) { - super.setValue(value, noChangedEvent) - this.setProperty("index", null) - } -} - -LiteGraph.registerNodeType({ - class: ComfyGalleryNode, - title: "UI.Gallery", - desc: "Gallery that shows most recent outputs", - type: "ui/gallery" -}) - -export interface ComfyButtonProperties extends ComfyWidgetProperties { - param: string -} - -export class ComfyButtonNode extends ComfyWidgetNode { - override properties: ComfyButtonProperties = { - tags: [], - defaultValue: false, - param: "bang" - } - - static slotLayout: SlotLayout = { - outputs: [ - { name: "clicked", type: BuiltInSlotType.EVENT }, - ] - } - - override svelteComponentType = ButtonWidget; - override defaultValue = false; - override outputSlotName = null; - override changedEventName = null; - - constructor(name?: string) { - super(name, false) - } - - override parseValue(param: any): boolean { - return Boolean(param); - } - - onClick() { - this.setValue(true) - this.triggerSlot(0, this.properties.param); - this.setValue(false) // TODO onRelease - } -} - -LiteGraph.registerNodeType({ - class: ComfyButtonNode, - title: "UI.Button", - desc: "Button that triggers an event when clicked", - type: "ui/button" -}) - -export interface ComfyCheckboxProperties extends ComfyWidgetProperties { -} - -export class ComfyCheckboxNode extends ComfyWidgetNode { - override properties: ComfyCheckboxProperties = { - tags: [], - defaultValue: false, - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "value", type: "boolean" }, - { name: "changed", type: BuiltInSlotType.EVENT }, - ] - } - - override svelteComponentType = CheckboxWidget; - override defaultValue = false; - - constructor(name?: string) { - super(name, false) - } - - override parseValue(param: any) { - return Boolean(param); - } -} - -LiteGraph.registerNodeType({ - class: ComfyCheckboxNode, - title: "UI.Checkbox", - desc: "Checkbox that stores a boolean value", - type: "ui/checkbox" -}) - -export interface ComfyRadioProperties extends ComfyWidgetProperties { - choices: string[] -} - -export class ComfyRadioNode extends ComfyWidgetNode { - override properties: ComfyRadioProperties = { - tags: [], - choices: ["Choice A", "Choice B", "Choice C"], - defaultValue: "Choice A", - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "value", type: "string" }, - { name: "index", type: "number" }, - { name: "changed", type: BuiltInSlotType.EVENT }, - ] - } - - override svelteComponentType = RadioWidget; - override defaultValue = ""; - - indexWidget: INumberWidget; - - index = 0; - - constructor(name?: string) { - super(name, "Choice A") - this.indexWidget = this.addWidget("number", "Index", this.index) - this.indexWidget.disabled = true; - } - - override onExecute(param: any, options: object) { - super.onExecute(param, options); - this.setOutputData(1, this.index) - } - - override setValue(value: string, noChangedEvent: boolean = false) { - super.setValue(value, noChangedEvent) - - value = get(this.value); - - const index = this.properties.choices.indexOf(value) - if (index === -1) - return; - - this.index = index; - this.indexWidget.value = index; - this.setOutputData(1, this.index) - } - - override parseValue(param: any): string { - if (typeof param === "string") { - if (this.properties.choices.indexOf(param) === -1) - return this.properties.choices[0] - return param - } - else { - const index = clamp(parseInt(param), 0, this.properties.choices.length - 1) - return this.properties.choices[index] || this.properties.defaultValue - } - } -} - -LiteGraph.registerNodeType({ - class: ComfyRadioNode, - title: "UI.Radio", - desc: "Radio that outputs a string and index", - type: "ui/radio" -}) - -export interface ComfyImageEditorNodeProperties extends ComfyWidgetProperties { -} - -export class ComfyImageEditorNode extends ComfyWidgetNode { - override properties: ComfyImageEditorNodeProperties = { - defaultValue: [], - tags: [], - } - - static slotLayout: SlotLayout = { - inputs: [ - { name: "store", type: BuiltInSlotType.ACTION } - ], - outputs: [ - { name: "images", type: "COMFYBOX_IMAGES" }, // TODO support batches - { name: "changed", type: BuiltInSlotType.EVENT }, - ] - } - - override svelteComponentType = ImageEditorWidget; - override defaultValue = []; - override storeActionName = "store"; - override saveUserState = false; - - constructor(name?: string) { - super(name, []) - } - - override parseValue(value: any): ComfyBoxImageMetadata[] { - return parseWhateverIntoImageMetadata(value) || []; - } - - override formatValue(value: GradioFileData[]): string { - return `Images: ${value?.length || 0}` - } -} - -LiteGraph.registerNodeType({ - class: ComfyImageEditorNode, - title: "UI.ImageEditor", - desc: "Widget that lets you upload and edit a multi-layered image. Can also act like a standalone image uploader.", - type: "ui/image_editor" -}) diff --git a/src/lib/nodes/index.ts b/src/lib/nodes/index.ts index 19b35cd..299314f 100644 --- a/src/lib/nodes/index.ts +++ b/src/lib/nodes/index.ts @@ -1,5 +1,4 @@ export { default as ComfyReroute } from "./ComfyReroute" -export { ComfyWidgetNode, ComfySliderNode, ComfyComboNode, ComfyTextNode } from "./ComfyWidgetNodes" export { ComfyQueueEvents, ComfyCopyAction, diff --git a/src/lib/utils.ts b/src/lib/utils.ts index 281c0e6..1a50088 100644 --- a/src/lib/utils.ts +++ b/src/lib/utils.ts @@ -7,7 +7,6 @@ import layoutState, { type WidgetLayout } from "$lib/stores/layoutState" import selectionState from "$lib/stores/selectionState" import type { SvelteComponentDev } from "svelte/internal"; import { Subgraph, type LGraph, type LGraphNode, type LLink, type SerializedLGraph, type UUID, GraphInput } from "@litegraph-ts/core"; -import type { ComfyExecutionResult, ComfyImageLocation } from "./nodes/ComfyWidgetNodes"; import type { FileData as GradioFileData } from "@gradio/upload"; import type { ComfyNodeID } from "./api"; @@ -332,6 +331,24 @@ export async function uploadImageToComfyUI(blob: Blob, filename: string, type: C }); } +/** Raw output as received from ComfyUI's backend */ +export interface ComfyExecutionResult { + // Technically this response can contain arbitrary data, but "images" is the + // most frequently used as it's output by LoadImage and PreviewImage, the + // only two output nodes in base ComfyUI. + images: ComfyImageLocation[] | null, +} + +/** Raw output entry as received from ComfyUI's backend */ +export type ComfyImageLocation = { + /* Filename with extension in the subfolder. */ + filename: string, + /* Subfolder in the containing folder. */ + subfolder: string, + /* Base ComfyUI folder where the image is located. */ + type: ComfyUploadImageType +} + /* * Convenient type for passing around image filepaths and their metadata with * wires. Needs to be converted to a filename for use with LoadImage. diff --git a/src/lib/widgets.ts b/src/lib/widgets.ts index e3083aa..8a67428 100644 --- a/src/lib/widgets.ts +++ b/src/lib/widgets.ts @@ -1,9 +1,7 @@ -import type { IWidget, LGraphNode } from "@litegraph-js/core"; -import ComfyValueControlWidget from "./widgets/ComfyValueControlWidget"; -import type { ComfyInputConfig } from "./IComfyInputSlot"; +import { LGraphNode, LiteGraph } from "@litegraph-ts/core"; import type IComfyInputSlot from "./IComfyInputSlot"; -import { BuiltInSlotShape, LiteGraph } from "@litegraph-ts/core"; -import { ComfyComboNode, ComfySliderNode, ComfyTextNode } from "./nodes"; +import type { ComfyInputConfig } from "./IComfyInputSlot"; +import { ComfyComboNode, ComfyNumberNode, ComfyTextNode } from "./nodes/widgets"; type WidgetFactory = (node: LGraphNode, inputName: string, inputData: any) => IComfyInputSlot; @@ -37,12 +35,12 @@ function addComfyInput(node: LGraphNode, inputName: string, extraInfo: Partial { const config = getNumberDefaults(inputData, 0.5); - return addComfyInput(node, inputName, { type: "number", config, defaultWidgetNode: ComfySliderNode }) + return addComfyInput(node, inputName, { type: "number", config, defaultWidgetNode: ComfyNumberNode }) } const INT: WidgetFactory = (node: LGraphNode, inputName: string, inputData: any): IComfyInputSlot => { const config = getNumberDefaults(inputData, 1); - return addComfyInput(node, inputName, { type: "number", config, defaultWidgetNode: ComfySliderNode }) + return addComfyInput(node, inputName, { type: "number", config, defaultWidgetNode: ComfyNumberNode }) }; const STRING: WidgetFactory = (node: LGraphNode, inputName: string, inputData: any): IComfyInputSlot => { diff --git a/src/lib/widgets/ButtonWidget.svelte b/src/lib/widgets/ButtonWidget.svelte index c92b6f5..7f5f1b1 100644 --- a/src/lib/widgets/ButtonWidget.svelte +++ b/src/lib/widgets/ButtonWidget.svelte @@ -1,10 +1,9 @@ + +
+ {#if widget.attrs.variant === "fileUpload" || isMobile} + + {:else} +
+ + +
+
+ +
+
+
+
+ + +
+ + + {#if !$nodeValue || $nodeValue.length === 0} + + + +
+ +
+ {#if uploadError} +
+ Upload error: {uploadError} +
+ {/if} +
+ + + + +
+ {:else} + + +
+ +
+ {#if uploadError} +
+ Upload error: {uploadError} +
+ {/if} +
+ {/if} +
+
+ {/if} +
+ + diff --git a/src/lib/widgets/NumberWidget.svelte b/src/lib/widgets/NumberWidget.svelte new file mode 100644 index 0000000..87c9294 --- /dev/null +++ b/src/lib/widgets/NumberWidget.svelte @@ -0,0 +1,173 @@ + + +
+ {#if node !== null && option !== null} + + {/if} +
+ + diff --git a/src/lib/widgets/RadioWidget.svelte b/src/lib/widgets/RadioWidget.svelte index c22eef9..f927d09 100644 --- a/src/lib/widgets/RadioWidget.svelte +++ b/src/lib/widgets/RadioWidget.svelte @@ -1,5 +1,4 @@ - -
- {#if node !== null && option !== null} - - {/if} -
- - diff --git a/src/lib/widgets/TextWidget.svelte b/src/lib/widgets/TextWidget.svelte index 79c8dbc..288602c 100644 --- a/src/lib/widgets/TextWidget.svelte +++ b/src/lib/widgets/TextWidget.svelte @@ -1,21 +1,21 @@