| | 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() { |
| | try { |
| | 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(); |
| | } |
| | catch(exception){ |
| | |
| | } |
| | } |
| |
|
| | 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(); |
| | app.resetView(); |
| | } |
| | } |
| | }), |
| | $el("button", { |
| | id: "comfy-load-default-button", textContent: "Load Default", onclick: async () => { |
| | if (!confirmClear.value || confirm("Load default workflow?")) { |
| | app.resetView(); |
| | await app.loadGraphData() |
| | } |
| | } |
| | }), |
| | $el("button", { |
| | id: "comfy-reset-view-button", textContent: "Reset View", onclick: async () => { |
| | app.resetView(); |
| | } |
| | }), |
| | ]); |
| |
|
| | 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; |
| | } |
| | } |
| | } |
| |
|