Spaces:
Running
on
Zero
Running
on
Zero
var __defProp = Object.defineProperty; | |
var __name = (target, value) => __defProp(target, "name", { value, configurable: true }); | |
import { e as LGraphNode, c as app, c3 as applyTextReplacements, c2 as ComfyWidgets, c5 as addValueControlWidgets, k as LiteGraph } from "./index-CoOvI8ZH.js"; | |
const CONVERTED_TYPE = "converted-widget"; | |
const VALID_TYPES = [ | |
"STRING", | |
"combo", | |
"number", | |
"toggle", | |
"BOOLEAN", | |
"text", | |
"string" | |
]; | |
const CONFIG = Symbol(); | |
const GET_CONFIG = Symbol(); | |
const TARGET = Symbol(); | |
const replacePropertyName = "Run widget replace on values"; | |
class PrimitiveNode extends LGraphNode { | |
static { | |
__name(this, "PrimitiveNode"); | |
} | |
controlValues; | |
lastType; | |
static category; | |
constructor(title) { | |
super(title); | |
this.addOutput("connect to widget input", "*"); | |
this.serialize_widgets = true; | |
this.isVirtualNode = true; | |
if (!this.properties || !(replacePropertyName in this.properties)) { | |
this.addProperty(replacePropertyName, false, "boolean"); | |
} | |
} | |
applyToGraph(extraLinks = []) { | |
if (!this.outputs[0].links?.length) return; | |
function get_links(node) { | |
let links2 = []; | |
for (const l of node.outputs[0].links) { | |
const linkInfo = app.graph.links[l]; | |
const n = node.graph.getNodeById(linkInfo.target_id); | |
if (n.type == "Reroute") { | |
links2 = links2.concat(get_links(n)); | |
} else { | |
links2.push(l); | |
} | |
} | |
return links2; | |
} | |
__name(get_links, "get_links"); | |
let links = [ | |
...get_links(this).map((l) => app.graph.links[l]), | |
...extraLinks | |
]; | |
let v = this.widgets?.[0].value; | |
if (v && this.properties[replacePropertyName]) { | |
v = applyTextReplacements(app, v); | |
} | |
for (const linkInfo of links) { | |
const node = this.graph.getNodeById(linkInfo.target_id); | |
const input = node.inputs[linkInfo.target_slot]; | |
let widget; | |
if (input.widget[TARGET]) { | |
widget = input.widget[TARGET]; | |
} else { | |
const widgetName = input.widget.name; | |
if (widgetName) { | |
widget = node.widgets.find((w) => w.name === widgetName); | |
} | |
} | |
if (widget) { | |
widget.value = v; | |
if (widget.callback) { | |
widget.callback( | |
widget.value, | |
app.canvas, | |
node, | |
app.canvas.graph_mouse, | |
{} | |
); | |
} | |
} | |
} | |
} | |
refreshComboInNode() { | |
const widget = this.widgets?.[0]; | |
if (widget?.type === "combo") { | |
widget.options.values = this.outputs[0].widget[GET_CONFIG]()[0]; | |
if (!widget.options.values.includes(widget.value)) { | |
widget.value = widget.options.values[0]; | |
widget.callback(widget.value); | |
} | |
} | |
} | |
onAfterGraphConfigured() { | |
if (this.outputs[0].links?.length && !this.widgets?.length) { | |
if (!this.#onFirstConnection()) return; | |
if (this.widgets) { | |
for (let i = 0; i < this.widgets_values.length; i++) { | |
const w = this.widgets[i]; | |
if (w) { | |
w.value = this.widgets_values[i]; | |
} | |
} | |
} | |
this.#mergeWidgetConfig(); | |
} | |
} | |
onConnectionsChange(_, index, connected) { | |
if (app.configuringGraph) { | |
return; | |
} | |
const links = this.outputs[0].links; | |
if (connected) { | |
if (links?.length && !this.widgets?.length) { | |
this.#onFirstConnection(); | |
} | |
} else { | |
this.#mergeWidgetConfig(); | |
if (!links?.length) { | |
this.onLastDisconnect(); | |
} | |
} | |
} | |
onConnectOutput(slot, type, input, target_node, target_slot) { | |
if (!input.widget) { | |
if (!(input.type in ComfyWidgets)) return false; | |
} | |
if (this.outputs[slot].links?.length) { | |
const valid = this.#isValidConnection(input); | |
if (valid) { | |
this.applyToGraph([{ target_id: target_node.id, target_slot }]); | |
} | |
return valid; | |
} | |
} | |
#onFirstConnection(recreating) { | |
if (!this.outputs[0].links) { | |
this.onLastDisconnect(); | |
return; | |
} | |
const linkId = this.outputs[0].links[0]; | |
const link = this.graph.links[linkId]; | |
if (!link) return; | |
const theirNode = this.graph.getNodeById(link.target_id); | |
if (!theirNode || !theirNode.inputs) return; | |
const input = theirNode.inputs[link.target_slot]; | |
if (!input) return; | |
let widget; | |
if (!input.widget) { | |
if (!(input.type in ComfyWidgets)) return; | |
widget = { name: input.name, [GET_CONFIG]: () => [input.type, {}] }; | |
} else { | |
widget = input.widget; | |
} | |
const config = widget[GET_CONFIG]?.(); | |
if (!config) return; | |
const { type } = getWidgetType(config); | |
this.outputs[0].type = type; | |
this.outputs[0].name = type; | |
this.outputs[0].widget = widget; | |
this.#createWidget( | |
widget[CONFIG] ?? config, | |
theirNode, | |
widget.name, | |
recreating, | |
widget[TARGET] | |
); | |
} | |
#createWidget(inputData, node, widgetName, recreating, targetWidget) { | |
let type = inputData[0]; | |
if (type instanceof Array) { | |
type = "COMBO"; | |
} | |
const [oldWidth, oldHeight] = this.size; | |
let widget; | |
if (type in ComfyWidgets) { | |
widget = (ComfyWidgets[type](this, "value", inputData, app) || {}).widget; | |
} else { | |
widget = this.addWidget(type, "value", null, () => { | |
}, {}); | |
} | |
if (targetWidget) { | |
widget.value = targetWidget.value; | |
} else if (node?.widgets && widget) { | |
const theirWidget = node.widgets.find((w) => w.name === widgetName); | |
if (theirWidget) { | |
widget.value = theirWidget.value; | |
} | |
} | |
if (!inputData?.[1]?.control_after_generate && (widget.type === "number" || widget.type === "combo")) { | |
let control_value = this.widgets_values?.[1]; | |
if (!control_value) { | |
control_value = "fixed"; | |
} | |
addValueControlWidgets( | |
this, | |
widget, | |
control_value, | |
void 0, | |
inputData | |
); | |
let filter = this.widgets_values?.[2]; | |
if (filter && this.widgets.length === 3) { | |
this.widgets[2].value = filter; | |
} | |
} | |
const controlValues = this.controlValues; | |
if (this.lastType === this.widgets[0].type && controlValues?.length === this.widgets.length - 1) { | |
for (let i = 0; i < controlValues.length; i++) { | |
this.widgets[i + 1].value = controlValues[i]; | |
} | |
} | |
const callback = widget.callback; | |
const self = this; | |
widget.callback = function() { | |
const r = callback ? callback.apply(this, arguments) : void 0; | |
self.applyToGraph(); | |
return r; | |
}; | |
this.size = [ | |
Math.max(this.size[0], oldWidth), | |
Math.max(this.size[1], oldHeight) | |
]; | |
if (!recreating) { | |
const sz = this.computeSize(); | |
if (this.size[0] < sz[0]) { | |
this.size[0] = sz[0]; | |
} | |
if (this.size[1] < sz[1]) { | |
this.size[1] = sz[1]; | |
} | |
requestAnimationFrame(() => { | |
if (this.onResize) { | |
this.onResize(this.size); | |
} | |
}); | |
} | |
} | |
recreateWidget() { | |
const values = this.widgets?.map((w) => w.value); | |
this.#removeWidgets(); | |
this.#onFirstConnection(true); | |
if (values?.length) { | |
for (let i = 0; i < this.widgets?.length; i++) | |
this.widgets[i].value = values[i]; | |
} | |
return this.widgets?.[0]; | |
} | |
#mergeWidgetConfig() { | |
const output = this.outputs[0]; | |
const links = output.links; | |
const hasConfig = !!output.widget[CONFIG]; | |
if (hasConfig) { | |
delete output.widget[CONFIG]; | |
} | |
if (links?.length < 2 && hasConfig) { | |
if (links.length) { | |
this.recreateWidget(); | |
} | |
return; | |
} | |
const config1 = output.widget[GET_CONFIG](); | |
const isNumber = config1[0] === "INT" || config1[0] === "FLOAT"; | |
if (!isNumber) return; | |
for (const linkId of links) { | |
const link = app.graph.links[linkId]; | |
if (!link) continue; | |
const theirNode = app.graph.getNodeById(link.target_id); | |
const theirInput = theirNode.inputs[link.target_slot]; | |
this.#isValidConnection(theirInput, hasConfig); | |
} | |
} | |
isValidWidgetLink(originSlot, targetNode, targetWidget) { | |
const config2 = getConfig.call(targetNode, targetWidget.name) ?? [ | |
targetWidget.type, | |
targetWidget.options || {} | |
]; | |
if (!isConvertibleWidget(targetWidget, config2)) return false; | |
const output = this.outputs[originSlot]; | |
if (!(output.widget?.[CONFIG] ?? output.widget?.[GET_CONFIG]())) { | |
return true; | |
} | |
return !!mergeIfValid.call(this, output, config2); | |
} | |
#isValidConnection(input, forceUpdate) { | |
const output = this.outputs[0]; | |
const config2 = input.widget[GET_CONFIG](); | |
return !!mergeIfValid.call( | |
this, | |
output, | |
config2, | |
forceUpdate, | |
this.recreateWidget | |
); | |
} | |
#removeWidgets() { | |
if (this.widgets) { | |
for (const w of this.widgets) { | |
if (w.onRemove) { | |
w.onRemove(); | |
} | |
} | |
this.controlValues = []; | |
this.lastType = this.widgets[0]?.type; | |
for (let i = 1; i < this.widgets.length; i++) { | |
this.controlValues.push(this.widgets[i].value); | |
} | |
setTimeout(() => { | |
delete this.lastType; | |
delete this.controlValues; | |
}, 15); | |
this.widgets.length = 0; | |
} | |
} | |
onLastDisconnect() { | |
this.outputs[0].type = "*"; | |
this.outputs[0].name = "connect to widget input"; | |
delete this.outputs[0].widget; | |
this.#removeWidgets(); | |
} | |
} | |
function getWidgetConfig(slot) { | |
return slot.widget[CONFIG] ?? slot.widget[GET_CONFIG]?.() ?? ["*", {}]; | |
} | |
__name(getWidgetConfig, "getWidgetConfig"); | |
function getConfig(widgetName) { | |
const { nodeData } = this.constructor; | |
return nodeData?.input?.required?.[widgetName] ?? nodeData?.input?.optional?.[widgetName]; | |
} | |
__name(getConfig, "getConfig"); | |
function isConvertibleWidget(widget, config) { | |
return (VALID_TYPES.includes(widget.type) || VALID_TYPES.includes(config[0])) && !widget.options?.forceInput; | |
} | |
__name(isConvertibleWidget, "isConvertibleWidget"); | |
function hideWidget(node, widget, suffix = "") { | |
if (widget.type?.startsWith(CONVERTED_TYPE)) return; | |
widget.origType = widget.type; | |
widget.origComputeSize = widget.computeSize; | |
widget.origSerializeValue = widget.serializeValue; | |
widget.computeSize = () => [0, -4]; | |
widget.type = CONVERTED_TYPE + suffix; | |
widget.serializeValue = () => { | |
if (!node.inputs) { | |
return void 0; | |
} | |
let node_input = node.inputs.find((i) => i.widget?.name === widget.name); | |
if (!node_input || !node_input.link) { | |
return void 0; | |
} | |
return widget.origSerializeValue ? widget.origSerializeValue() : widget.value; | |
}; | |
if (widget.linkedWidgets) { | |
for (const w of widget.linkedWidgets) { | |
hideWidget(node, w, ":" + widget.name); | |
} | |
} | |
} | |
__name(hideWidget, "hideWidget"); | |
function showWidget(widget) { | |
widget.type = widget.origType; | |
widget.computeSize = widget.origComputeSize; | |
widget.serializeValue = widget.origSerializeValue; | |
delete widget.origType; | |
delete widget.origComputeSize; | |
delete widget.origSerializeValue; | |
if (widget.linkedWidgets) { | |
for (const w of widget.linkedWidgets) { | |
showWidget(w); | |
} | |
} | |
} | |
__name(showWidget, "showWidget"); | |
function convertToInput(node, widget, config) { | |
hideWidget(node, widget); | |
const { type } = getWidgetType(config); | |
const [oldWidth, oldHeight] = node.size; | |
const inputIsOptional = !!widget.options?.inputIsOptional; | |
const input = node.addInput(widget.name, type, { | |
widget: { name: widget.name, [GET_CONFIG]: () => config }, | |
...inputIsOptional ? { shape: LiteGraph.SlotShape.HollowCircle } : {} | |
}); | |
for (const widget2 of node.widgets) { | |
widget2.last_y += LiteGraph.NODE_SLOT_HEIGHT; | |
} | |
node.setSize([ | |
Math.max(oldWidth, node.size[0]), | |
Math.max(oldHeight, node.size[1]) | |
]); | |
return input; | |
} | |
__name(convertToInput, "convertToInput"); | |
function convertToWidget(node, widget) { | |
showWidget(widget); | |
const [oldWidth, oldHeight] = node.size; | |
node.removeInput(node.inputs.findIndex((i) => i.widget?.name === widget.name)); | |
for (const widget2 of node.widgets) { | |
widget2.last_y -= LiteGraph.NODE_SLOT_HEIGHT; | |
} | |
node.setSize([ | |
Math.max(oldWidth, node.size[0]), | |
Math.max(oldHeight, node.size[1]) | |
]); | |
} | |
__name(convertToWidget, "convertToWidget"); | |
function getWidgetType(config) { | |
let type = config[0]; | |
if (type instanceof Array) { | |
type = "COMBO"; | |
} | |
return { type }; | |
} | |
__name(getWidgetType, "getWidgetType"); | |
function isValidCombo(combo, obj) { | |
if (!(obj instanceof Array)) { | |
console.log(`connection rejected: tried to connect combo to ${obj}`); | |
return false; | |
} | |
if (combo.length !== obj.length) { | |
console.log(`connection rejected: combo lists dont match`); | |
return false; | |
} | |
if (combo.find((v, i) => obj[i] !== v)) { | |
console.log(`connection rejected: combo lists dont match`); | |
return false; | |
} | |
return true; | |
} | |
__name(isValidCombo, "isValidCombo"); | |
function isPrimitiveNode(node) { | |
return node.type === "PrimitiveNode"; | |
} | |
__name(isPrimitiveNode, "isPrimitiveNode"); | |
function setWidgetConfig(slot, config, target) { | |
if (!slot.widget) return; | |
if (config) { | |
slot.widget[GET_CONFIG] = () => config; | |
slot.widget[TARGET] = target; | |
} else { | |
delete slot.widget; | |
} | |
if (slot.link) { | |
const link = app.graph.links[slot.link]; | |
if (link) { | |
const originNode = app.graph.getNodeById(link.origin_id); | |
if (isPrimitiveNode(originNode)) { | |
if (config) { | |
originNode.recreateWidget(); | |
} else if (!app.configuringGraph) { | |
originNode.disconnectOutput(0); | |
originNode.onLastDisconnect(); | |
} | |
} | |
} | |
} | |
} | |
__name(setWidgetConfig, "setWidgetConfig"); | |
function mergeIfValid(output, config2, forceUpdate, recreateWidget, config1) { | |
if (!config1) { | |
config1 = getWidgetConfig(output); | |
} | |
if (config1[0] instanceof Array) { | |
if (!isValidCombo(config1[0], config2[0])) return; | |
} else if (config1[0] !== config2[0]) { | |
console.log(`connection rejected: types dont match`, config1[0], config2[0]); | |
return; | |
} | |
const keys = /* @__PURE__ */ new Set([ | |
...Object.keys(config1[1] ?? {}), | |
...Object.keys(config2[1] ?? {}) | |
]); | |
let customConfig; | |
const getCustomConfig = /* @__PURE__ */ __name(() => { | |
if (!customConfig) { | |
if (typeof structuredClone === "undefined") { | |
customConfig = JSON.parse(JSON.stringify(config1[1] ?? {})); | |
} else { | |
customConfig = structuredClone(config1[1] ?? {}); | |
} | |
} | |
return customConfig; | |
}, "getCustomConfig"); | |
const isNumber = config1[0] === "INT" || config1[0] === "FLOAT"; | |
for (const k of keys.values()) { | |
if (k !== "default" && k !== "forceInput" && k !== "defaultInput" && k !== "control_after_generate" && k !== "multiline" && k !== "tooltip") { | |
let v1 = config1[1][k]; | |
let v2 = config2[1]?.[k]; | |
if (v1 === v2 || !v1 && !v2) continue; | |
if (isNumber) { | |
if (k === "min") { | |
const theirMax = config2[1]?.["max"]; | |
if (theirMax != null && v1 > theirMax) { | |
console.log("connection rejected: min > max", v1, theirMax); | |
return; | |
} | |
getCustomConfig()[k] = v1 == null ? v2 : v2 == null ? v1 : Math.max(v1, v2); | |
continue; | |
} else if (k === "max") { | |
const theirMin = config2[1]?.["min"]; | |
if (theirMin != null && v1 < theirMin) { | |
console.log("connection rejected: max < min", v1, theirMin); | |
return; | |
} | |
getCustomConfig()[k] = v1 == null ? v2 : v2 == null ? v1 : Math.min(v1, v2); | |
continue; | |
} else if (k === "step") { | |
let step; | |
if (v1 == null) { | |
step = v2; | |
} else if (v2 == null) { | |
step = v1; | |
} else { | |
if (v1 < v2) { | |
const a = v2; | |
v2 = v1; | |
v1 = a; | |
} | |
if (v1 % v2) { | |
console.log( | |
"connection rejected: steps not divisible", | |
"current:", | |
v1, | |
"new:", | |
v2 | |
); | |
return; | |
} | |
step = v1; | |
} | |
getCustomConfig()[k] = step; | |
continue; | |
} | |
} | |
console.log(`connection rejected: config ${k} values dont match`, v1, v2); | |
return; | |
} | |
} | |
if (customConfig || forceUpdate) { | |
if (customConfig) { | |
output.widget[CONFIG] = [config1[0], customConfig]; | |
} | |
const widget = recreateWidget?.call(this); | |
if (widget) { | |
const min = widget.options.min; | |
const max = widget.options.max; | |
if (min != null && widget.value < min) widget.value = min; | |
if (max != null && widget.value > max) widget.value = max; | |
widget.callback(widget.value); | |
} | |
} | |
return { customConfig }; | |
} | |
__name(mergeIfValid, "mergeIfValid"); | |
let useConversionSubmenusSetting; | |
app.registerExtension({ | |
name: "Comfy.WidgetInputs", | |
init() { | |
useConversionSubmenusSetting = app.ui.settings.addSetting({ | |
id: "Comfy.NodeInputConversionSubmenus", | |
name: "In the node context menu, place the entries that convert between input/widget in sub-menus.", | |
type: "boolean", | |
defaultValue: true | |
}); | |
}, | |
async beforeRegisterNodeDef(nodeType, nodeData, app2) { | |
const origGetExtraMenuOptions = nodeType.prototype.getExtraMenuOptions; | |
nodeType.prototype.convertWidgetToInput = function(widget) { | |
const config = getConfig.call(this, widget.name) ?? [ | |
widget.type, | |
widget.options || {} | |
]; | |
if (!isConvertibleWidget(widget, config)) return false; | |
if (widget.type?.startsWith(CONVERTED_TYPE)) return false; | |
convertToInput(this, widget, config); | |
return true; | |
}; | |
nodeType.prototype.getExtraMenuOptions = function(_, options) { | |
const r = origGetExtraMenuOptions ? origGetExtraMenuOptions.apply(this, arguments) : void 0; | |
if (this.widgets) { | |
let toInput = []; | |
let toWidget = []; | |
for (const w of this.widgets) { | |
if (w.options?.forceInput) { | |
continue; | |
} | |
if (w.type === CONVERTED_TYPE) { | |
toWidget.push({ | |
content: `Convert ${w.name} to widget`, | |
callback: /* @__PURE__ */ __name(() => convertToWidget(this, w), "callback") | |
}); | |
} else { | |
const config = getConfig.call(this, w.name) ?? [ | |
w.type, | |
w.options || {} | |
]; | |
if (isConvertibleWidget(w, config)) { | |
toInput.push({ | |
content: `Convert ${w.name} to input`, | |
callback: /* @__PURE__ */ __name(() => convertToInput(this, w, config), "callback") | |
}); | |
} | |
} | |
} | |
if (toInput.length) { | |
if (useConversionSubmenusSetting.value) { | |
options.push({ | |
content: "Convert Widget to Input", | |
submenu: { | |
options: toInput | |
} | |
}); | |
} else { | |
options.push(...toInput, null); | |
} | |
} | |
if (toWidget.length) { | |
if (useConversionSubmenusSetting.value) { | |
options.push({ | |
content: "Convert Input to Widget", | |
submenu: { | |
options: toWidget | |
} | |
}); | |
} else { | |
options.push(...toWidget, null); | |
} | |
} | |
} | |
return r; | |
}; | |
nodeType.prototype.onGraphConfigured = function() { | |
if (!this.inputs) return; | |
this.widgets ??= []; | |
for (const input of this.inputs) { | |
if (input.widget) { | |
if (!input.widget[GET_CONFIG]) { | |
input.widget[GET_CONFIG] = () => getConfig.call(this, input.widget.name); | |
} | |
if (input.widget.config) { | |
if (input.widget.config[0] instanceof Array) { | |
input.type = "COMBO"; | |
const link = app2.graph.links[input.link]; | |
if (link) { | |
link.type = input.type; | |
} | |
} | |
delete input.widget.config; | |
} | |
const w = this.widgets.find((w2) => w2.name === input.widget.name); | |
if (w) { | |
hideWidget(this, w); | |
} else { | |
convertToWidget(this, input); | |
} | |
} | |
} | |
}; | |
const origOnNodeCreated = nodeType.prototype.onNodeCreated; | |
nodeType.prototype.onNodeCreated = function() { | |
const r = origOnNodeCreated ? origOnNodeCreated.apply(this) : void 0; | |
if (!app2.configuringGraph && this.widgets) { | |
for (const w of this.widgets) { | |
if (w?.options?.forceInput || w?.options?.defaultInput) { | |
const config = getConfig.call(this, w.name) ?? [ | |
w.type, | |
w.options || {} | |
]; | |
convertToInput(this, w, config); | |
} | |
} | |
} | |
return r; | |
}; | |
const origOnConfigure = nodeType.prototype.onConfigure; | |
nodeType.prototype.onConfigure = function() { | |
const r = origOnConfigure ? origOnConfigure.apply(this, arguments) : void 0; | |
if (!app2.configuringGraph && this.inputs) { | |
for (const input of this.inputs) { | |
if (input.widget && !input.widget[GET_CONFIG]) { | |
input.widget[GET_CONFIG] = () => getConfig.call(this, input.widget.name); | |
const w = this.widgets.find((w2) => w2.name === input.widget.name); | |
if (w) { | |
hideWidget(this, w); | |
} | |
} | |
} | |
} | |
return r; | |
}; | |
function isNodeAtPos(pos) { | |
for (const n of app2.graph.nodes) { | |
if (n.pos[0] === pos[0] && n.pos[1] === pos[1]) { | |
return true; | |
} | |
} | |
return false; | |
} | |
__name(isNodeAtPos, "isNodeAtPos"); | |
const origOnInputDblClick = nodeType.prototype.onInputDblClick; | |
const ignoreDblClick = Symbol(); | |
nodeType.prototype.onInputDblClick = function(slot) { | |
const r = origOnInputDblClick ? origOnInputDblClick.apply(this, arguments) : void 0; | |
const input = this.inputs[slot]; | |
if (!input.widget || !input[ignoreDblClick]) { | |
if (!(input.type in ComfyWidgets) && !(input.widget?.[GET_CONFIG]?.()?.[0] instanceof Array)) { | |
return r; | |
} | |
} | |
const node = LiteGraph.createNode("PrimitiveNode"); | |
app2.graph.add(node); | |
const pos = [ | |
this.pos[0] - node.size[0] - 30, | |
this.pos[1] | |
]; | |
while (isNodeAtPos(pos)) { | |
pos[1] += LiteGraph.NODE_TITLE_HEIGHT; | |
} | |
node.pos = pos; | |
node.connect(0, this, slot); | |
node.title = input.name; | |
input[ignoreDblClick] = true; | |
setTimeout(() => { | |
delete input[ignoreDblClick]; | |
}, 300); | |
return r; | |
}; | |
const onConnectInput = nodeType.prototype.onConnectInput; | |
nodeType.prototype.onConnectInput = function(targetSlot, type, output, originNode, originSlot) { | |
const v = onConnectInput?.(this, arguments); | |
if (type !== "COMBO") return v; | |
if (originNode.outputs[originSlot].widget) return v; | |
const targetCombo = this.inputs[targetSlot].widget?.[GET_CONFIG]?.()?.[0]; | |
if (!targetCombo || !(targetCombo instanceof Array)) return v; | |
const originConfig = originNode.constructor?.nodeData?.output?.[originSlot]; | |
if (!originConfig || !isValidCombo(targetCombo, originConfig)) { | |
return false; | |
} | |
return v; | |
}; | |
}, | |
registerCustomNodes() { | |
LiteGraph.registerNodeType( | |
"PrimitiveNode", | |
Object.assign(PrimitiveNode, { | |
title: "Primitive" | |
}) | |
); | |
PrimitiveNode.category = "utils"; | |
} | |
}); | |
window.comfyAPI = window.comfyAPI || {}; | |
window.comfyAPI.widgetInputs = window.comfyAPI.widgetInputs || {}; | |
window.comfyAPI.widgetInputs.getWidgetConfig = getWidgetConfig; | |
window.comfyAPI.widgetInputs.convertToInput = convertToInput; | |
window.comfyAPI.widgetInputs.setWidgetConfig = setWidgetConfig; | |
window.comfyAPI.widgetInputs.mergeIfValid = mergeIfValid; | |
export { | |
convertToInput, | |
getWidgetConfig, | |
mergeIfValid, | |
setWidgetConfig | |
}; | |
//# sourceMappingURL=widgetInputs-CRPRgKEi.js.map | |