| | import { api } from "./api.js" |
| | import "./domWidget.js"; |
| |
|
| | let controlValueRunBefore = false; |
| | export function updateControlWidgetLabel(widget) { |
| | let replacement = "after"; |
| | let find = "before"; |
| | if (controlValueRunBefore) { |
| | [find, replacement] = [replacement, find] |
| | } |
| | widget.label = (widget.label ?? widget.name).replace(find, replacement); |
| | } |
| |
|
| | const IS_CONTROL_WIDGET = Symbol(); |
| | const HAS_EXECUTED = Symbol(); |
| |
|
| | function getNumberDefaults(inputData, defaultStep, precision, enable_rounding) { |
| | let defaultVal = inputData[1]["default"]; |
| | let { min, max, step, round} = inputData[1]; |
| |
|
| | if (defaultVal == undefined) defaultVal = 0; |
| | if (min == undefined) min = 0; |
| | if (max == undefined) max = 2048; |
| | if (step == undefined) step = defaultStep; |
| | |
| | |
| | if (precision == undefined) { |
| | precision = Math.max(-Math.floor(Math.log10(step)),0); |
| | } |
| |
|
| | if (enable_rounding && (round == undefined || round === true)) { |
| | |
| | round = Math.round(1000000*Math.pow(0.1,precision))/1000000; |
| | } |
| |
|
| | return { val: defaultVal, config: { min, max, step: 10.0 * step, round, precision } }; |
| | } |
| |
|
| | export function addValueControlWidget(node, targetWidget, defaultValue = "randomize", values, widgetName, inputData) { |
| | let name = inputData[1]?.control_after_generate; |
| | if(typeof name !== "string") { |
| | name = widgetName; |
| | } |
| | const widgets = addValueControlWidgets(node, targetWidget, defaultValue, { |
| | addFilterList: false, |
| | controlAfterGenerateName: name |
| | }, inputData); |
| | return widgets[0]; |
| | } |
| |
|
| | export function addValueControlWidgets(node, targetWidget, defaultValue = "randomize", options, inputData) { |
| | if (!defaultValue) defaultValue = "randomize"; |
| | if (!options) options = {}; |
| |
|
| | const getName = (defaultName, optionName) => { |
| | let name = defaultName; |
| | if (options[optionName]) { |
| | name = options[optionName]; |
| | } else if (typeof inputData?.[1]?.[defaultName] === "string") { |
| | name = inputData?.[1]?.[defaultName]; |
| | } else if (inputData?.[1]?.control_prefix) { |
| | name = inputData?.[1]?.control_prefix + " " + name |
| | } |
| | return name; |
| | } |
| |
|
| | const widgets = []; |
| | const valueControl = node.addWidget( |
| | "combo", |
| | getName("control_after_generate", "controlAfterGenerateName"), |
| | defaultValue, |
| | function () {}, |
| | { |
| | values: ["fixed", "increment", "decrement", "randomize"], |
| | serialize: false, |
| | } |
| | ); |
| | valueControl[IS_CONTROL_WIDGET] = true; |
| | updateControlWidgetLabel(valueControl); |
| | widgets.push(valueControl); |
| |
|
| | const isCombo = targetWidget.type === "combo"; |
| | let comboFilter; |
| | if (isCombo) { |
| | valueControl.options.values.push("increment-wrap"); |
| | } |
| | if (isCombo && options.addFilterList !== false) { |
| | comboFilter = node.addWidget( |
| | "string", |
| | getName("control_filter_list", "controlFilterListName"), |
| | "", |
| | function () {}, |
| | { |
| | serialize: false, |
| | } |
| | ); |
| | updateControlWidgetLabel(comboFilter); |
| |
|
| | widgets.push(comboFilter); |
| | } |
| |
|
| | const applyWidgetControl = () => { |
| | var v = valueControl.value; |
| |
|
| | if (isCombo && v !== "fixed") { |
| | let values = targetWidget.options.values; |
| | const filter = comboFilter?.value; |
| | if (filter) { |
| | let check; |
| | if (filter.startsWith("/") && filter.endsWith("/")) { |
| | try { |
| | const regex = new RegExp(filter.substring(1, filter.length - 1)); |
| | check = (item) => regex.test(item); |
| | } catch (error) { |
| | console.error("Error constructing RegExp filter for node " + node.id, filter, error); |
| | } |
| | } |
| | if (!check) { |
| | const lower = filter.toLocaleLowerCase(); |
| | check = (item) => item.toLocaleLowerCase().includes(lower); |
| | } |
| | values = values.filter(item => check(item)); |
| | if (!values.length && targetWidget.options.values.length) { |
| | console.warn("Filter for node " + node.id + " has filtered out all items", filter); |
| | } |
| | } |
| | let current_index = values.indexOf(targetWidget.value); |
| | let current_length = values.length; |
| |
|
| | switch (v) { |
| | case "increment": |
| | current_index += 1; |
| | break; |
| | case "increment-wrap": |
| | current_index += 1; |
| | if ( current_index >= current_length ) { |
| | current_index = 0; |
| | } |
| | break; |
| | case "decrement": |
| | current_index -= 1; |
| | break; |
| | case "randomize": |
| | current_index = Math.floor(Math.random() * current_length); |
| | default: |
| | break; |
| | } |
| | current_index = Math.max(0, current_index); |
| | current_index = Math.min(current_length - 1, current_index); |
| | if (current_index >= 0) { |
| | let value = values[current_index]; |
| | targetWidget.value = value; |
| | targetWidget.callback(value); |
| | } |
| | } else { |
| | |
| | let min = targetWidget.options.min; |
| | let max = targetWidget.options.max; |
| | |
| | max = Math.min(1125899906842624, max); |
| | min = Math.max(-1125899906842624, min); |
| | let range = (max - min) / (targetWidget.options.step / 10); |
| |
|
| | |
| | switch (v) { |
| | case "fixed": |
| | break; |
| | case "increment": |
| | targetWidget.value += targetWidget.options.step / 10; |
| | break; |
| | case "decrement": |
| | targetWidget.value -= targetWidget.options.step / 10; |
| | break; |
| | case "randomize": |
| | targetWidget.value = Math.floor(Math.random() * range) * (targetWidget.options.step / 10) + min; |
| | default: |
| | break; |
| | } |
| | |
| | |
| | if (targetWidget.value < min) targetWidget.value = min; |
| |
|
| | if (targetWidget.value > max) |
| | targetWidget.value = max; |
| | targetWidget.callback(targetWidget.value); |
| | } |
| | }; |
| |
|
| | valueControl.beforeQueued = () => { |
| | if (controlValueRunBefore) { |
| | |
| | if (valueControl[HAS_EXECUTED]) { |
| | applyWidgetControl(); |
| | } |
| | } |
| | valueControl[HAS_EXECUTED] = true; |
| | }; |
| |
|
| | valueControl.afterQueued = () => { |
| | if (!controlValueRunBefore) { |
| | applyWidgetControl(); |
| | } |
| | }; |
| |
|
| | return widgets; |
| | }; |
| |
|
| | function seedWidget(node, inputName, inputData, app, widgetName) { |
| | const seed = createIntWidget(node, inputName, inputData, app, true); |
| | const seedControl = addValueControlWidget(node, seed.widget, "randomize", undefined, widgetName, inputData); |
| |
|
| | seed.widget.linkedWidgets = [seedControl]; |
| | return seed; |
| | } |
| |
|
| | function createIntWidget(node, inputName, inputData, app, isSeedInput) { |
| | const control = inputData[1]?.control_after_generate; |
| | if (!isSeedInput && control) { |
| | return seedWidget(node, inputName, inputData, app, typeof control === "string" ? control : undefined); |
| | } |
| |
|
| | let widgetType = isSlider(inputData[1]["display"], app); |
| | const { val, config } = getNumberDefaults(inputData, 1, 0, true); |
| | Object.assign(config, { precision: 0 }); |
| | return { |
| | widget: node.addWidget( |
| | widgetType, |
| | inputName, |
| | val, |
| | function (v) { |
| | const s = this.options.step / 10; |
| | let sh = this.options.min % s; |
| | if (isNaN(sh)) { |
| | sh = 0; |
| | } |
| | this.value = Math.round((v - sh) / s) * s + sh; |
| | }, |
| | config |
| | ), |
| | }; |
| | } |
| |
|
| | function addMultilineWidget(node, name, opts, app) { |
| | const inputEl = document.createElement("textarea"); |
| | inputEl.className = "comfy-multiline-input"; |
| | inputEl.value = opts.defaultVal; |
| | inputEl.placeholder = opts.placeholder || name; |
| |
|
| | const widget = node.addDOMWidget(name, "customtext", inputEl, { |
| | getValue() { |
| | return inputEl.value; |
| | }, |
| | setValue(v) { |
| | inputEl.value = v; |
| | }, |
| | }); |
| | widget.inputEl = inputEl; |
| |
|
| | inputEl.addEventListener("input", () => { |
| | widget.callback?.(widget.value); |
| | }); |
| |
|
| | return { minWidth: 400, minHeight: 200, widget }; |
| | } |
| |
|
| | function isSlider(display, app) { |
| | if (app.ui.settings.getSettingValue("Comfy.DisableSliders")) { |
| | return "number" |
| | } |
| |
|
| | return (display==="slider") ? "slider" : "number" |
| | } |
| |
|
| | export function initWidgets(app) { |
| | app.ui.settings.addSetting({ |
| | id: "Comfy.WidgetControlMode", |
| | name: "Widget Value Control Mode", |
| | type: "combo", |
| | defaultValue: "after", |
| | options: ["before", "after"], |
| | tooltip: "Controls when widget values are updated (randomize/increment/decrement), either before the prompt is queued or after.", |
| | onChange(value) { |
| | controlValueRunBefore = value === "before"; |
| | for (const n of app.graph._nodes) { |
| | if (!n.widgets) continue; |
| | for (const w of n.widgets) { |
| | if (w[IS_CONTROL_WIDGET]) { |
| | updateControlWidgetLabel(w); |
| | if (w.linkedWidgets) { |
| | for (const l of w.linkedWidgets) { |
| | updateControlWidgetLabel(l); |
| | } |
| | } |
| | } |
| | } |
| | } |
| | app.graph.setDirtyCanvas(true); |
| | }, |
| | }); |
| | } |
| |
|
| | export const ComfyWidgets = { |
| | "INT:seed": seedWidget, |
| | "INT:noise_seed": seedWidget, |
| | FLOAT(node, inputName, inputData, app) { |
| | let widgetType = isSlider(inputData[1]["display"], app); |
| | let precision = app.ui.settings.getSettingValue("Comfy.FloatRoundingPrecision"); |
| | let disable_rounding = app.ui.settings.getSettingValue("Comfy.DisableFloatRounding") |
| | if (precision == 0) precision = undefined; |
| | const { val, config } = getNumberDefaults(inputData, 0.5, precision, !disable_rounding); |
| | return { widget: node.addWidget(widgetType, inputName, val, |
| | function (v) { |
| | if (config.round) { |
| | this.value = Math.round((v + Number.EPSILON)/config.round)*config.round; |
| | if (this.value > config.max) this.value = config.max; |
| | if (this.value < config.min) this.value = config.min; |
| | } else { |
| | this.value = v; |
| | } |
| | }, config) }; |
| | }, |
| | INT(node, inputName, inputData, app) { |
| | return createIntWidget(node, inputName, inputData, app); |
| | }, |
| | BOOLEAN(node, inputName, inputData) { |
| | let defaultVal = false; |
| | let options = {}; |
| | if (inputData[1]) { |
| | if (inputData[1].default) |
| | defaultVal = inputData[1].default; |
| | if (inputData[1].label_on) |
| | options["on"] = inputData[1].label_on; |
| | if (inputData[1].label_off) |
| | options["off"] = inputData[1].label_off; |
| | } |
| | return { |
| | widget: node.addWidget( |
| | "toggle", |
| | inputName, |
| | defaultVal, |
| | () => {}, |
| | options, |
| | ) |
| | }; |
| | }, |
| | STRING(node, inputName, inputData, app) { |
| | const defaultVal = inputData[1].default || ""; |
| | const multiline = !!inputData[1].multiline; |
| |
|
| | let res; |
| | if (multiline) { |
| | res = addMultilineWidget(node, inputName, { defaultVal, ...inputData[1] }, app); |
| | } else { |
| | res = { widget: node.addWidget("text", inputName, defaultVal, () => {}, {}) }; |
| | } |
| |
|
| | if(inputData[1].dynamicPrompts != undefined) |
| | res.widget.dynamicPrompts = inputData[1].dynamicPrompts; |
| |
|
| | return res; |
| | }, |
| | COMBO(node, inputName, inputData) { |
| | const type = inputData[0]; |
| | let defaultValue = type[0]; |
| | if (inputData[1] && inputData[1].default) { |
| | defaultValue = inputData[1].default; |
| | } |
| | const res = { widget: node.addWidget("combo", inputName, defaultValue, () => {}, { values: type }) }; |
| | if (inputData[1]?.control_after_generate) { |
| | res.widget.linkedWidgets = addValueControlWidgets(node, res.widget, undefined, undefined, inputData); |
| | } |
| | return res; |
| | }, |
| | IMAGEUPLOAD(node, inputName, inputData, app) { |
| | const imageWidget = node.widgets.find((w) => w.name === (inputData[1]?.widget ?? "image")); |
| | let uploadWidget; |
| |
|
| | function showImage(name) { |
| | const img = new Image(); |
| | img.onload = () => { |
| | node.imgs = [img]; |
| | app.graph.setDirtyCanvas(true); |
| | }; |
| | let folder_separator = name.lastIndexOf("/"); |
| | let subfolder = ""; |
| | if (folder_separator > -1) { |
| | subfolder = name.substring(0, folder_separator); |
| | name = name.substring(folder_separator + 1); |
| | } |
| | img.src = api.apiURL(`/view?filename=${encodeURIComponent(name)}&type=input&subfolder=${subfolder}${app.getPreviewFormatParam()}${app.getRandParam()}`); |
| | node.setSizeForImage?.(); |
| | } |
| |
|
| | var default_value = imageWidget.value; |
| | Object.defineProperty(imageWidget, "value", { |
| | set : function(value) { |
| | this._real_value = value; |
| | }, |
| |
|
| | get : function() { |
| | let value = ""; |
| | if (this._real_value) { |
| | value = this._real_value; |
| | } else { |
| | return default_value; |
| | } |
| |
|
| | if (value.filename) { |
| | let real_value = value; |
| | value = ""; |
| | if (real_value.subfolder) { |
| | value = real_value.subfolder + "/"; |
| | } |
| |
|
| | value += real_value.filename; |
| |
|
| | if(real_value.type && real_value.type !== "input") |
| | value += ` [${real_value.type}]`; |
| | } |
| | return value; |
| | } |
| | }); |
| |
|
| | |
| | const cb = node.callback; |
| | imageWidget.callback = function () { |
| | showImage(imageWidget.value); |
| | if (cb) { |
| | return cb.apply(this, arguments); |
| | } |
| | }; |
| |
|
| | |
| | |
| | |
| | requestAnimationFrame(() => { |
| | if (imageWidget.value) { |
| | showImage(imageWidget.value); |
| | } |
| | }); |
| |
|
| | async function uploadFile(file, updateNode, pasted = false) { |
| | try { |
| | |
| | const body = new FormData(); |
| | body.append("image", file); |
| | if (pasted) body.append("subfolder", "pasted"); |
| | const resp = await api.fetchApi("/upload/image", { |
| | method: "POST", |
| | body, |
| | }); |
| |
|
| | if (resp.status === 200) { |
| | const data = await resp.json(); |
| | |
| | let path = data.name; |
| | if (data.subfolder) path = data.subfolder + "/" + path; |
| |
|
| | if (!imageWidget.options.values.includes(path)) { |
| | imageWidget.options.values.push(path); |
| | } |
| |
|
| | if (updateNode) { |
| | showImage(path); |
| | imageWidget.value = path; |
| | } |
| | } else { |
| | alert(resp.status + " - " + resp.statusText); |
| | } |
| | } catch (error) { |
| | alert(error); |
| | } |
| | } |
| |
|
| | const fileInput = document.createElement("input"); |
| | Object.assign(fileInput, { |
| | type: "file", |
| | accept: "image/jpeg,image/png,image/webp", |
| | style: "display: none", |
| | onchange: async () => { |
| | if (fileInput.files.length) { |
| | await uploadFile(fileInput.files[0], true); |
| | } |
| | }, |
| | }); |
| | document.body.append(fileInput); |
| |
|
| | |
| | uploadWidget = node.addWidget("button", inputName, "image", () => { |
| | fileInput.click(); |
| | }); |
| | uploadWidget.label = "choose file to upload"; |
| | uploadWidget.serialize = false; |
| |
|
| | |
| | node.onDragOver = function (e) { |
| | if (e.dataTransfer && e.dataTransfer.items) { |
| | const image = [...e.dataTransfer.items].find((f) => f.kind === "file"); |
| | return !!image; |
| | } |
| |
|
| | return false; |
| | }; |
| |
|
| | |
| | node.onDragDrop = function (e) { |
| | console.log("onDragDrop called"); |
| | let handled = false; |
| | for (const file of e.dataTransfer.files) { |
| | if (file.type.startsWith("image/")) { |
| | uploadFile(file, !handled); |
| | handled = true; |
| | } |
| | } |
| |
|
| | return handled; |
| | }; |
| |
|
| | node.pasteFile = function(file) { |
| | if (file.type.startsWith("image/")) { |
| | const is_pasted = (file.name === "image.png") && |
| | (file.lastModified - Date.now() < 2000); |
| | uploadFile(file, true, is_pasted); |
| | return true; |
| | } |
| | return false; |
| | } |
| |
|
| | return { widget: uploadWidget }; |
| | }, |
| | }; |
| |
|