| | import {api} from "./api.js"; |
| |
|
| | 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 (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); |
| | } |
| |
|
| | 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; |
| | } |
| | } |
| |
|
| | export class ComfyDialog { |
| | constructor() { |
| | this.element = $el("div.comfy-modal", {parent: document.body}, [ |
| | $el("div.comfy-modal-content", [$el("p", {$: (p) => (this.textElement = p)}), ...this.createButtons()]), |
| | ]); |
| | } |
| |
|
| | createButtons() { |
| | return [ |
| | $el("button", { |
| | type: "button", |
| | textContent: "Close", |
| | onclick: () => this.close(), |
| | }), |
| | ]; |
| | } |
| |
|
| | close() { |
| | this.element.style.display = "none"; |
| | } |
| |
|
| | show(html) { |
| | if (typeof html === "string") { |
| | this.textElement.innerHTML = html; |
| | } else { |
| | this.textElement.replaceChildren(html); |
| | } |
| | this.element.style.display = "flex"; |
| | } |
| | } |
| |
|
| | class ComfySettingsDialog extends ComfyDialog { |
| | constructor() { |
| | super(); |
| | this.element = $el("dialog", { |
| | id: "comfy-settings-dialog", |
| | parent: document.body, |
| | }, [ |
| | $el("table.comfy-modal-content.comfy-table", [ |
| | $el("caption", {textContent: "Settings"}), |
| | $el("tbody", {$: (tbody) => (this.textElement = tbody)}), |
| | $el("button", { |
| | type: "button", |
| | textContent: "Close", |
| | style: { |
| | cursor: "pointer", |
| | }, |
| | onclick: () => { |
| | this.element.close(); |
| | }, |
| | }), |
| | ]), |
| | ]); |
| | this.settings = []; |
| | } |
| |
|
| | getSettingValue(id, defaultValue) { |
| | const settingId = "Comfy.Settings." + id; |
| | const v = localStorage[settingId]; |
| | return v == null ? defaultValue : JSON.parse(v); |
| | } |
| |
|
| | setSettingValue(id, value) { |
| | const settingId = "Comfy.Settings." + id; |
| | localStorage[settingId] = JSON.stringify(value); |
| | } |
| |
|
| | addSetting({id, name, type, defaultValue, onChange, attrs = {}, tooltip = "", options = undefined}) { |
| | if (!id) { |
| | throw new Error("Settings must have an ID"); |
| | } |
| |
|
| | if (this.settings.find((s) => s.id === id)) { |
| | throw new Error(`Setting ${id} of type ${type} must have a unique ID.`); |
| | } |
| |
|
| | const settingId = `Comfy.Settings.${id}`; |
| | const v = localStorage[settingId]; |
| | let value = v == null ? defaultValue : JSON.parse(v); |
| |
|
| | |
| | if (onChange) { |
| | onChange(value, undefined); |
| | } |
| |
|
| | this.settings.push({ |
| | render: () => { |
| | const setter = (v) => { |
| | if (onChange) { |
| | onChange(v, value); |
| | } |
| | localStorage[settingId] = JSON.stringify(v); |
| | value = v; |
| | }; |
| | value = this.getSettingValue(id, defaultValue); |
| |
|
| | let element; |
| | const htmlID = id.replaceAll(".", "-"); |
| |
|
| | const labelCell = $el("td", [ |
| | $el("label", { |
| | for: htmlID, |
| | classList: [tooltip !== "" ? "comfy-tooltip-indicator" : ""], |
| | textContent: name, |
| | }) |
| | ]); |
| |
|
| | if (typeof type === "function") { |
| | element = type(name, setter, value, attrs); |
| | } else { |
| | switch (type) { |
| | case "boolean": |
| | element = $el("tr", [ |
| | labelCell, |
| | $el("td", [ |
| | $el("input", { |
| | id: htmlID, |
| | type: "checkbox", |
| | checked: value, |
| | onchange: (event) => { |
| | const isChecked = event.target.checked; |
| | if (onChange !== undefined) { |
| | onChange(isChecked) |
| | } |
| | this.setSettingValue(id, isChecked); |
| | }, |
| | }), |
| | ]), |
| | ]) |
| | break; |
| | case "number": |
| | element = $el("tr", [ |
| | labelCell, |
| | $el("td", [ |
| | $el("input", { |
| | type, |
| | value, |
| | id: htmlID, |
| | oninput: (e) => { |
| | setter(e.target.value); |
| | }, |
| | ...attrs |
| | }), |
| | ]), |
| | ]); |
| | break; |
| | case "slider": |
| | element = $el("tr", [ |
| | labelCell, |
| | $el("td", [ |
| | $el("div", { |
| | style: { |
| | display: "grid", |
| | gridAutoFlow: "column", |
| | }, |
| | }, [ |
| | $el("input", { |
| | ...attrs, |
| | value, |
| | type: "range", |
| | oninput: (e) => { |
| | setter(e.target.value); |
| | e.target.nextElementSibling.value = e.target.value; |
| | }, |
| | }), |
| | $el("input", { |
| | ...attrs, |
| | value, |
| | id: htmlID, |
| | type: "number", |
| | style: {maxWidth: "4rem"}, |
| | oninput: (e) => { |
| | setter(e.target.value); |
| | e.target.previousElementSibling.value = e.target.value; |
| | }, |
| | }), |
| | ]), |
| | ]), |
| | ]); |
| | break; |
| | case "combo": |
| | element = $el("tr", [ |
| | labelCell, |
| | $el("td", [ |
| | $el( |
| | "select", |
| | { |
| | oninput: (e) => { |
| | setter(e.target.value); |
| | }, |
| | }, |
| | (typeof options === "function" ? options(value) : options || []).map((opt) => { |
| | if (typeof opt === "string") { |
| | opt = { text: opt }; |
| | } |
| | const v = opt.value ?? opt.text; |
| | return $el("option", { |
| | value: v, |
| | textContent: opt.text, |
| | selected: value + "" === v + "", |
| | }); |
| | }) |
| | ), |
| | ]), |
| | ]); |
| | break; |
| | case "text": |
| | default: |
| | if (type !== "text") { |
| | console.warn(`Unsupported setting type '${type}, defaulting to text`); |
| | } |
| |
|
| | element = $el("tr", [ |
| | labelCell, |
| | $el("td", [ |
| | $el("input", { |
| | value, |
| | id: htmlID, |
| | oninput: (e) => { |
| | setter(e.target.value); |
| | }, |
| | ...attrs, |
| | }), |
| | ]), |
| | ]); |
| | break; |
| | } |
| | } |
| | if (tooltip) { |
| | element.title = tooltip; |
| | } |
| |
|
| | return element; |
| | }, |
| | }); |
| |
|
| | const self = this; |
| | return { |
| | get value() { |
| | return self.getSettingValue(id, defaultValue); |
| | }, |
| | set value(v) { |
| | self.setSettingValue(id, v); |
| | }, |
| | }; |
| | } |
| |
|
| | show() { |
| | this.textElement.replaceChildren( |
| | $el("tr", { |
| | style: {display: "none"}, |
| | }, [ |
| | $el("th"), |
| | $el("th", {style: {width: "33%"}}) |
| | ]), |
| | ...this.settings.map((s) => s.render()), |
| | ) |
| | this.element.showModal(); |
| | } |
| | } |
| |
|
| | 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: () => { |
| | 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(); |
| |
|
| | 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", |
| | style: {display: "none"}, |
| | parent: document.body, |
| | onchange: () => { |
| | app.handleFile(fileInput.files[0]); |
| | }, |
| | }); |
| |
|
| | this.menuContainer = $el("div.comfy-menu", {parent: document.body}, [ |
| | $el("div.drag-handle", { |
| | style: { |
| | overflow: "hidden", |
| | position: "relative", |
| | width: "100%", |
| | cursor: "default" |
| | } |
| | }, [ |
| | $el("span.drag-handle"), |
| | $el("span", {$: (q) => (this.queueSize = q)}), |
| | $el("button.comfy-settings-btn", {textContent: "⚙️", onclick: () => this.settings.show()}), |
| | ]), |
| | $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; |
| | }, |
| | }), |
| | ]), |
| | ]), |
| | $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", |
| | |
| | }), |
| | ]) |
| | ]), |
| | $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"; |
| | } |
| | } |
| | const json = JSON.stringify(app.graph.serialize(), 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: () => { |
| | if (!confirmClear.value || confirm("Load default workflow?")) { |
| | 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 && |
| | document.getElementById("autoQueueCheckbox").checked |
| | ) { |
| | app.queuePrompt(0, this.batchCount); |
| | } |
| | this.lastQueueSize = status.exec_info.queue_remaining; |
| | } |
| | } |
| | } |
| |
|