|
import { api } from "./api.js"; |
|
import { ComfyDialog as _ComfyDialog } from "./ui/dialog.js"; |
|
import { toggleSwitch } from "./ui/toggleSwitch.js"; |
|
import { ComfySettingsDialog } from "./ui/settings.js"; |
|
|
|
export const ComfyDialog = _ComfyDialog; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function $el(tag, propsOrChildren, children) { |
|
const split = tag.split("."); |
|
const element = document.createElement(split.shift()); |
|
if (split.length > 0) { |
|
element.classList.add(...split); |
|
} |
|
|
|
if (propsOrChildren) { |
|
if (typeof propsOrChildren === "string") { |
|
propsOrChildren = { textContent: propsOrChildren }; |
|
} else if (propsOrChildren instanceof Element) { |
|
propsOrChildren = [propsOrChildren]; |
|
} |
|
if (Array.isArray(propsOrChildren)) { |
|
element.append(...propsOrChildren); |
|
} else { |
|
const {parent, $: cb, dataset, style} = propsOrChildren; |
|
delete propsOrChildren.parent; |
|
delete propsOrChildren.$; |
|
delete propsOrChildren.dataset; |
|
delete propsOrChildren.style; |
|
|
|
if (Object.hasOwn(propsOrChildren, "for")) { |
|
element.setAttribute("for", propsOrChildren.for) |
|
} |
|
|
|
if (style) { |
|
Object.assign(element.style, style); |
|
} |
|
|
|
if (dataset) { |
|
Object.assign(element.dataset, dataset); |
|
} |
|
|
|
Object.assign(element, propsOrChildren); |
|
if (children) { |
|
element.append(...(children instanceof Array ? children : [children])); |
|
} |
|
|
|
if (parent) { |
|
parent.append(element); |
|
} |
|
|
|
if (cb) { |
|
cb(element); |
|
} |
|
} |
|
} |
|
return element; |
|
} |
|
|
|
function dragElement(dragEl, settings) { |
|
var posDiffX = 0, |
|
posDiffY = 0, |
|
posStartX = 0, |
|
posStartY = 0, |
|
newPosX = 0, |
|
newPosY = 0; |
|
if (dragEl.getElementsByClassName("drag-handle")[0]) { |
|
|
|
dragEl.getElementsByClassName("drag-handle")[0].onmousedown = dragMouseDown; |
|
} else { |
|
|
|
dragEl.onmousedown = dragMouseDown; |
|
} |
|
|
|
|
|
const resizeObserver = new ResizeObserver(() => { |
|
ensureInBounds(); |
|
}).observe(dragEl); |
|
|
|
function ensureInBounds() { |
|
if (dragEl.classList.contains("comfy-menu-manual-pos")) { |
|
newPosX = Math.min(document.body.clientWidth - dragEl.clientWidth, Math.max(0, dragEl.offsetLeft)); |
|
newPosY = Math.min(document.body.clientHeight - dragEl.clientHeight, Math.max(0, dragEl.offsetTop)); |
|
|
|
positionElement(); |
|
} |
|
} |
|
|
|
function positionElement() { |
|
const halfWidth = document.body.clientWidth / 2; |
|
const anchorRight = newPosX + dragEl.clientWidth / 2 > halfWidth; |
|
|
|
|
|
if (anchorRight) { |
|
dragEl.style.left = "unset"; |
|
dragEl.style.right = document.body.clientWidth - newPosX - dragEl.clientWidth + "px"; |
|
} else { |
|
dragEl.style.left = newPosX + "px"; |
|
dragEl.style.right = "unset"; |
|
} |
|
|
|
dragEl.style.top = newPosY + "px"; |
|
dragEl.style.bottom = "unset"; |
|
|
|
if (savePos) { |
|
localStorage.setItem( |
|
"Comfy.MenuPosition", |
|
JSON.stringify({ |
|
x: dragEl.offsetLeft, |
|
y: dragEl.offsetTop, |
|
}) |
|
); |
|
} |
|
} |
|
|
|
function restorePos() { |
|
let pos = localStorage.getItem("Comfy.MenuPosition"); |
|
if (pos) { |
|
pos = JSON.parse(pos); |
|
newPosX = pos.x; |
|
newPosY = pos.y; |
|
positionElement(); |
|
ensureInBounds(); |
|
} |
|
} |
|
|
|
let savePos = undefined; |
|
settings.addSetting({ |
|
id: "Comfy.MenuPosition", |
|
name: "Save menu position", |
|
type: "boolean", |
|
defaultValue: savePos, |
|
onChange(value) { |
|
if (savePos === undefined && value) { |
|
restorePos(); |
|
} |
|
savePos = value; |
|
}, |
|
}); |
|
|
|
function dragMouseDown(e) { |
|
e = e || window.event; |
|
e.preventDefault(); |
|
|
|
posStartX = e.clientX; |
|
posStartY = e.clientY; |
|
document.onmouseup = closeDragElement; |
|
|
|
document.onmousemove = elementDrag; |
|
} |
|
|
|
function elementDrag(e) { |
|
e = e || window.event; |
|
e.preventDefault(); |
|
|
|
dragEl.classList.add("comfy-menu-manual-pos"); |
|
|
|
|
|
posDiffX = e.clientX - posStartX; |
|
posDiffY = e.clientY - posStartY; |
|
posStartX = e.clientX; |
|
posStartY = e.clientY; |
|
|
|
newPosX = Math.min(document.body.clientWidth - dragEl.clientWidth, Math.max(0, dragEl.offsetLeft + posDiffX)); |
|
newPosY = Math.min(document.body.clientHeight - dragEl.clientHeight, Math.max(0, dragEl.offsetTop + posDiffY)); |
|
|
|
positionElement(); |
|
} |
|
|
|
window.addEventListener("resize", () => { |
|
ensureInBounds(); |
|
}); |
|
|
|
function closeDragElement() { |
|
|
|
document.onmouseup = null; |
|
document.onmousemove = null; |
|
} |
|
} |
|
|
|
class ComfyList { |
|
#type; |
|
#text; |
|
#reverse; |
|
|
|
constructor(text, type, reverse) { |
|
this.#text = text; |
|
this.#type = type || text.toLowerCase(); |
|
this.#reverse = reverse || false; |
|
this.element = $el("div.comfy-list"); |
|
this.element.style.display = "none"; |
|
} |
|
|
|
get visible() { |
|
return this.element.style.display !== "none"; |
|
} |
|
|
|
async load() { |
|
const items = await api.getItems(this.#type); |
|
this.element.replaceChildren( |
|
...Object.keys(items).flatMap((section) => [ |
|
$el("h4", { |
|
textContent: section, |
|
}), |
|
$el("div.comfy-list-items", [ |
|
...(this.#reverse ? items[section].reverse() : items[section]).map((item) => { |
|
|
|
const removeAction = item.remove || { |
|
name: "Delete", |
|
cb: () => api.deleteItem(this.#type, item.prompt[1]), |
|
}; |
|
return $el("div", {textContent: item.prompt[0] + ": "}, [ |
|
$el("button", { |
|
textContent: "Load", |
|
onclick: async () => { |
|
await app.loadGraphData(item.prompt[3].extra_pnginfo.workflow); |
|
if (item.outputs) { |
|
app.nodeOutputs = item.outputs; |
|
} |
|
}, |
|
}), |
|
$el("button", { |
|
textContent: removeAction.name, |
|
onclick: async () => { |
|
await removeAction.cb(); |
|
await this.update(); |
|
}, |
|
}), |
|
]); |
|
}), |
|
]), |
|
]), |
|
$el("div.comfy-list-actions", [ |
|
$el("button", { |
|
textContent: "Clear " + this.#text, |
|
onclick: async () => { |
|
await api.clearItems(this.#type); |
|
await this.load(); |
|
}, |
|
}), |
|
$el("button", {textContent: "Refresh", onclick: () => this.load()}), |
|
]) |
|
); |
|
} |
|
|
|
async update() { |
|
if (this.visible) { |
|
await this.load(); |
|
} |
|
} |
|
|
|
async show() { |
|
this.element.style.display = "block"; |
|
this.button.textContent = "Close"; |
|
|
|
await this.load(); |
|
} |
|
|
|
hide() { |
|
this.element.style.display = "none"; |
|
this.button.textContent = "View " + this.#text; |
|
} |
|
|
|
toggle() { |
|
if (this.visible) { |
|
this.hide(); |
|
return false; |
|
} else { |
|
this.show(); |
|
return true; |
|
} |
|
} |
|
} |
|
|
|
export class ComfyUI { |
|
constructor(app) { |
|
this.app = app; |
|
this.dialog = new ComfyDialog(); |
|
this.settings = new ComfySettingsDialog(app); |
|
|
|
this.batchCount = 1; |
|
this.lastQueueSize = 0; |
|
this.queue = new ComfyList("Queue"); |
|
this.history = new ComfyList("History", "history", true); |
|
|
|
api.addEventListener("status", () => { |
|
this.queue.update(); |
|
this.history.update(); |
|
}); |
|
|
|
const confirmClear = this.settings.addSetting({ |
|
id: "Comfy.ConfirmClear", |
|
name: "Require confirmation when clearing workflow", |
|
type: "boolean", |
|
defaultValue: true, |
|
}); |
|
|
|
const promptFilename = this.settings.addSetting({ |
|
id: "Comfy.PromptFilename", |
|
name: "Prompt for filename when saving workflow", |
|
type: "boolean", |
|
defaultValue: true, |
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const previewImage = this.settings.addSetting({ |
|
id: "Comfy.PreviewFormat", |
|
name: "When displaying a preview in the image widget, convert it to a lightweight image, e.g. webp, jpeg, webp;50, etc.", |
|
type: "text", |
|
defaultValue: "", |
|
}); |
|
|
|
this.settings.addSetting({ |
|
id: "Comfy.DisableSliders", |
|
name: "Disable sliders.", |
|
type: "boolean", |
|
defaultValue: false, |
|
}); |
|
|
|
this.settings.addSetting({ |
|
id: "Comfy.DisableFloatRounding", |
|
name: "Disable rounding floats (requires page reload).", |
|
type: "boolean", |
|
defaultValue: false, |
|
}); |
|
|
|
this.settings.addSetting({ |
|
id: "Comfy.FloatRoundingPrecision", |
|
name: "Decimal places [0 = auto] (requires page reload).", |
|
type: "slider", |
|
attrs: { |
|
min: 0, |
|
max: 6, |
|
step: 1, |
|
}, |
|
defaultValue: 0, |
|
}); |
|
|
|
const fileInput = $el("input", { |
|
id: "comfy-file-input", |
|
type: "file", |
|
accept: ".json,image/png,.latent,.safetensors,image/webp", |
|
style: {display: "none"}, |
|
parent: document.body, |
|
onchange: () => { |
|
app.handleFile(fileInput.files[0]); |
|
}, |
|
}); |
|
|
|
const autoQueueModeEl = toggleSwitch( |
|
"autoQueueMode", |
|
[ |
|
{ text: "instant", tooltip: "A new prompt will be queued as soon as the queue reaches 0" }, |
|
{ text: "change", tooltip: "A new prompt will be queued when the queue is at 0 and the graph is/has changed" }, |
|
], |
|
{ |
|
onChange: (value) => { |
|
this.autoQueueMode = value.item.value; |
|
}, |
|
} |
|
); |
|
autoQueueModeEl.style.display = "none"; |
|
|
|
api.addEventListener("graphChanged", () => { |
|
if (this.autoQueueMode === "change" && this.autoQueueEnabled === true) { |
|
if (this.lastQueueSize === 0) { |
|
this.graphHasChanged = false; |
|
app.queuePrompt(0, this.batchCount); |
|
} else { |
|
this.graphHasChanged = true; |
|
} |
|
} |
|
}); |
|
|
|
this.menuHamburger = $el( |
|
"div.comfy-menu-hamburger", |
|
{ |
|
parent: document.body, |
|
onclick: () => { |
|
this.menuContainer.style.display = "block"; |
|
this.menuHamburger.style.display = "none"; |
|
}, |
|
}, |
|
[$el("div"), $el("div"), $el("div")] |
|
); |
|
|
|
this.menuContainer = $el("div.comfy-menu", { parent: document.body }, [ |
|
$el("div.drag-handle.comfy-menu-header", { |
|
style: { |
|
overflow: "hidden", |
|
position: "relative", |
|
width: "100%", |
|
cursor: "default" |
|
} |
|
}, [ |
|
$el("span.drag-handle"), |
|
$el("span.comfy-menu-queue-size", { $: (q) => (this.queueSize = q) }), |
|
$el("div.comfy-menu-actions", [ |
|
$el("button.comfy-settings-btn", { |
|
textContent: "⚙️", |
|
onclick: () => this.settings.show(), |
|
}), |
|
$el("button.comfy-close-menu-btn", { |
|
textContent: "\u00d7", |
|
onclick: () => { |
|
this.menuContainer.style.display = "none"; |
|
this.menuHamburger.style.display = "flex"; |
|
}, |
|
}), |
|
]), |
|
]), |
|
$el("button.comfy-queue-btn", { |
|
id: "queue-button", |
|
textContent: "Queue Prompt", |
|
onclick: () => app.queuePrompt(0, this.batchCount), |
|
}), |
|
$el("div", {}, [ |
|
$el("label", {innerHTML: "Extra options"}, [ |
|
$el("input", { |
|
type: "checkbox", |
|
onchange: (i) => { |
|
document.getElementById("extraOptions").style.display = i.srcElement.checked ? "block" : "none"; |
|
this.batchCount = i.srcElement.checked ? document.getElementById("batchCountInputRange").value : 1; |
|
document.getElementById("autoQueueCheckbox").checked = false; |
|
this.autoQueueEnabled = false; |
|
}, |
|
}), |
|
]), |
|
]), |
|
$el("div", {id: "extraOptions", style: {width: "100%", display: "none"}}, [ |
|
$el("div",[ |
|
|
|
$el("label", {innerHTML: "Batch count"}), |
|
$el("input", { |
|
id: "batchCountInputNumber", |
|
type: "number", |
|
value: this.batchCount, |
|
min: "1", |
|
style: {width: "35%", "margin-left": "0.4em"}, |
|
oninput: (i) => { |
|
this.batchCount = i.target.value; |
|
document.getElementById("batchCountInputRange").value = this.batchCount; |
|
}, |
|
}), |
|
$el("input", { |
|
id: "batchCountInputRange", |
|
type: "range", |
|
min: "1", |
|
max: "100", |
|
value: this.batchCount, |
|
oninput: (i) => { |
|
this.batchCount = i.srcElement.value; |
|
document.getElementById("batchCountInputNumber").value = i.srcElement.value; |
|
}, |
|
}), |
|
]), |
|
$el("div",[ |
|
$el("label",{ |
|
for:"autoQueueCheckbox", |
|
innerHTML: "Auto Queue" |
|
}), |
|
$el("input", { |
|
id: "autoQueueCheckbox", |
|
type: "checkbox", |
|
checked: false, |
|
title: "Automatically queue prompt when the queue size hits 0", |
|
onchange: (e) => { |
|
this.autoQueueEnabled = e.target.checked; |
|
autoQueueModeEl.style.display = this.autoQueueEnabled ? "" : "none"; |
|
} |
|
}), |
|
autoQueueModeEl |
|
]) |
|
]), |
|
$el("div.comfy-menu-btns", [ |
|
$el("button", { |
|
id: "queue-front-button", |
|
textContent: "Queue Front", |
|
onclick: () => app.queuePrompt(-1, this.batchCount) |
|
}), |
|
$el("button", { |
|
$: (b) => (this.queue.button = b), |
|
id: "comfy-view-queue-button", |
|
textContent: "View Queue", |
|
onclick: () => { |
|
this.history.hide(); |
|
this.queue.toggle(); |
|
}, |
|
}), |
|
$el("button", { |
|
$: (b) => (this.history.button = b), |
|
id: "comfy-view-history-button", |
|
textContent: "View History", |
|
onclick: () => { |
|
this.queue.hide(); |
|
this.history.toggle(); |
|
}, |
|
}), |
|
]), |
|
this.queue.element, |
|
this.history.element, |
|
$el("button", { |
|
id: "comfy-save-button", |
|
textContent: "Save", |
|
onclick: () => { |
|
let filename = "workflow.json"; |
|
if (promptFilename.value) { |
|
filename = prompt("Save workflow as:", filename); |
|
if (!filename) return; |
|
if (!filename.toLowerCase().endsWith(".json")) { |
|
filename += ".json"; |
|
} |
|
} |
|
app.graphToPrompt().then(p=>{ |
|
const json = JSON.stringify(p.workflow, null, 2); |
|
const blob = new Blob([json], {type: "application/json"}); |
|
const url = URL.createObjectURL(blob); |
|
const a = $el("a", { |
|
href: url, |
|
download: filename, |
|
style: {display: "none"}, |
|
parent: document.body, |
|
}); |
|
a.click(); |
|
setTimeout(function () { |
|
a.remove(); |
|
window.URL.revokeObjectURL(url); |
|
}, 0); |
|
}); |
|
}, |
|
}), |
|
$el("button", { |
|
id: "comfy-dev-save-api-button", |
|
textContent: "Save (API Format)", |
|
style: {width: "100%", display: "none"}, |
|
onclick: () => { |
|
let filename = "workflow_api.json"; |
|
if (promptFilename.value) { |
|
filename = prompt("Save workflow (API) as:", filename); |
|
if (!filename) return; |
|
if (!filename.toLowerCase().endsWith(".json")) { |
|
filename += ".json"; |
|
} |
|
} |
|
app.graphToPrompt().then(p=>{ |
|
const json = JSON.stringify(p.output, null, 2); |
|
const blob = new Blob([json], {type: "application/json"}); |
|
const url = URL.createObjectURL(blob); |
|
const a = $el("a", { |
|
href: url, |
|
download: filename, |
|
style: {display: "none"}, |
|
parent: document.body, |
|
}); |
|
a.click(); |
|
setTimeout(function () { |
|
a.remove(); |
|
window.URL.revokeObjectURL(url); |
|
}, 0); |
|
}); |
|
}, |
|
}), |
|
$el("button", {id: "comfy-load-button", textContent: "Load", onclick: () => fileInput.click()}), |
|
$el("button", { |
|
id: "comfy-refresh-button", |
|
textContent: "Refresh", |
|
onclick: () => app.refreshComboInNodes() |
|
}), |
|
$el("button", {id: "comfy-clipspace-button", textContent: "Clipspace", onclick: () => app.openClipspace()}), |
|
$el("button", { |
|
id: "comfy-clear-button", textContent: "Clear", onclick: () => { |
|
if (!confirmClear.value || confirm("Clear workflow?")) { |
|
app.clean(); |
|
app.graph.clear(); |
|
} |
|
} |
|
}), |
|
$el("button", { |
|
id: "comfy-load-default-button", textContent: "Load Default", onclick: async () => { |
|
if (!confirmClear.value || confirm("Load default workflow?")) { |
|
await app.loadGraphData() |
|
} |
|
} |
|
}), |
|
]); |
|
|
|
const devMode = this.settings.addSetting({ |
|
id: "Comfy.DevMode", |
|
name: "Enable Dev mode Options", |
|
type: "boolean", |
|
defaultValue: false, |
|
onChange: function(value) { document.getElementById("comfy-dev-save-api-button").style.display = value ? "block" : "none"}, |
|
}); |
|
|
|
dragElement(this.menuContainer, this.settings); |
|
|
|
this.setStatus({exec_info: {queue_remaining: "X"}}); |
|
} |
|
|
|
setStatus(status) { |
|
this.queueSize.textContent = "Queue size: " + (status ? status.exec_info.queue_remaining : "ERR"); |
|
if (status) { |
|
if ( |
|
this.lastQueueSize != 0 && |
|
status.exec_info.queue_remaining == 0 && |
|
this.autoQueueEnabled && |
|
(this.autoQueueMode === "instant" || this.graphHasChanged) && |
|
!app.lastExecutionError |
|
) { |
|
app.queuePrompt(0, this.batchCount); |
|
status.exec_info.queue_remaining += this.batchCount; |
|
this.graphHasChanged = false; |
|
} |
|
this.lastQueueSize = status.exec_info.queue_remaining; |
|
} |
|
} |
|
} |
|
|