|
import { app } from "../../scripts/app.js"; |
|
import { api } from "../../scripts/api.js" |
|
import { sleep, show_message } from "./common.js"; |
|
import { GroupNodeConfig, GroupNodeHandler } from "../../extensions/core/groupNode.js"; |
|
import { ComfyDialog, $el } from "../../scripts/ui.js"; |
|
|
|
const SEPARATOR = ">" |
|
|
|
let pack_map = {}; |
|
let rpack_map = {}; |
|
|
|
export function getPureName(node) { |
|
|
|
let category = null; |
|
if(node.category) { |
|
category = node.category.substring(12); |
|
} |
|
else { |
|
category = node.constructor.category?.substring(12); |
|
} |
|
if(category) { |
|
let purename = node.comfyClass.substring(category.length+1); |
|
return purename; |
|
} |
|
else if(node.comfyClass.startsWith('workflow/') || node.comfyClass.startsWith(`workflow${SEPARATOR}`)) { |
|
return node.comfyClass.substring(9); |
|
} |
|
else { |
|
return node.comfyClass; |
|
} |
|
} |
|
|
|
function isValidVersionString(version) { |
|
const versionPattern = /^(\d+)\.(\d+)(\.(\d+))?$/; |
|
|
|
const match = version.match(versionPattern); |
|
|
|
return match !== null && |
|
parseInt(match[1], 10) >= 0 && |
|
parseInt(match[2], 10) >= 0 && |
|
(!match[3] || parseInt(match[4], 10) >= 0); |
|
} |
|
|
|
function register_pack_map(name, data) { |
|
if(data.packname) { |
|
pack_map[data.packname] = name; |
|
rpack_map[name] = data; |
|
} |
|
else { |
|
rpack_map[name] = data; |
|
} |
|
} |
|
|
|
function storeGroupNode(name, data, register=true) { |
|
let extra = app.graph.extra; |
|
if (!extra) app.graph.extra = extra = {}; |
|
let groupNodes = extra.groupNodes; |
|
if (!groupNodes) extra.groupNodes = groupNodes = {}; |
|
groupNodes[name] = data; |
|
|
|
if(register) { |
|
register_pack_map(name, data); |
|
} |
|
} |
|
|
|
export async function load_components() { |
|
let data = await api.fetchApi('/manager/component/loads', {method: "POST"}); |
|
let components = await data.json(); |
|
|
|
let start_time = Date.now(); |
|
let failed = []; |
|
let failed2 = []; |
|
|
|
for(let name in components) { |
|
if(app.graph.extra?.groupNodes?.[name]) { |
|
if(data) { |
|
let data = components[name]; |
|
|
|
let category = data.packname; |
|
if(data.category) { |
|
category += SEPARATOR + data.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
const config = new GroupNodeConfig(name, data); |
|
await config.registerType(category); |
|
|
|
register_pack_map(name, data); |
|
continue; |
|
} |
|
} |
|
|
|
let nodeData = components[name]; |
|
|
|
storeGroupNode(name, nodeData); |
|
|
|
const config = new GroupNodeConfig(name, nodeData); |
|
|
|
while(true) { |
|
try { |
|
let category = nodeData.packname; |
|
if(nodeData.category) { |
|
category += SEPARATOR + nodeData.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
await config.registerType(category); |
|
register_pack_map(name, nodeData); |
|
break; |
|
} |
|
catch { |
|
let elapsed_time = Date.now() - start_time; |
|
if (elapsed_time > 5000) { |
|
failed.push(name); |
|
break; |
|
} else { |
|
await sleep(100); |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
for(let i in failed) { |
|
let name = failed[i]; |
|
|
|
if(app.graph.extra?.groupNodes?.[name]) { |
|
continue; |
|
} |
|
|
|
let nodeData = components[name]; |
|
|
|
storeGroupNode(name, nodeData); |
|
|
|
const config = new GroupNodeConfig(name, nodeData); |
|
while(true) { |
|
try { |
|
let category = nodeData.packname; |
|
if(nodeData.workflow.category) { |
|
category += SEPARATOR + nodeData.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
await config.registerType(category); |
|
register_pack_map(name, nodeData); |
|
break; |
|
} |
|
catch { |
|
let elapsed_time = Date.now() - start_time; |
|
if (elapsed_time > 10000) { |
|
failed2.push(name); |
|
break; |
|
} else { |
|
await sleep(100); |
|
} |
|
} |
|
} |
|
} |
|
|
|
|
|
for(let name in failed2) { |
|
let name = failed2[i]; |
|
|
|
let nodeData = components[name]; |
|
|
|
storeGroupNode(name, nodeData); |
|
|
|
const config = new GroupNodeConfig(name, nodeData); |
|
while(true) { |
|
try { |
|
let category = nodeData.workflow.packname; |
|
if(nodeData.workflow.category) { |
|
category += SEPARATOR + nodeData.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
await config.registerType(category); |
|
register_pack_map(name, nodeData); |
|
break; |
|
} |
|
catch { |
|
let elapsed_time = Date.now() - start_time; |
|
if (elapsed_time > 30000) { |
|
failed.push(name); |
|
break; |
|
} else { |
|
await sleep(100); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
|
|
async function save_as_component(node, version, author, prefix, nodename, packname, category) { |
|
let component_name = `${prefix}::${nodename}`; |
|
|
|
let subgraph = app.graph.extra?.groupNodes?.[component_name]; |
|
if(!subgraph) { |
|
subgraph = app.graph.extra?.groupNodes?.[getPureName(node)]; |
|
} |
|
|
|
subgraph.version = version; |
|
subgraph.author = author; |
|
subgraph.datetime = Date.now(); |
|
subgraph.packname = packname; |
|
subgraph.category = category; |
|
|
|
let body = |
|
{ |
|
name: component_name, |
|
workflow: subgraph |
|
}; |
|
|
|
pack_map[packname] = component_name; |
|
rpack_map[component_name] = subgraph; |
|
|
|
const res = await api.fetchApi('/manager/component/save', { |
|
method: "POST", |
|
headers: { |
|
"Content-Type": "application/json", |
|
}, |
|
body: JSON.stringify(body), |
|
}); |
|
|
|
if(res.status == 200) { |
|
storeGroupNode(component_name, subgraph); |
|
const config = new GroupNodeConfig(component_name, subgraph); |
|
|
|
let category = body.workflow.packname; |
|
if(body.workflow.category) { |
|
category += SEPARATOR + body.workflow.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
await config.registerType(category); |
|
|
|
let path = await res.text(); |
|
show_message(`Component '${component_name}' is saved into:\n${path}`); |
|
} |
|
else |
|
show_message(`Failed to save component.`); |
|
} |
|
|
|
async function import_component(component_name, component, mode) { |
|
if(mode) { |
|
let body = |
|
{ |
|
name: component_name, |
|
workflow: component |
|
}; |
|
|
|
const res = await api.fetchApi('/manager/component/save', { |
|
method: "POST", |
|
headers: { "Content-Type": "application/json", }, |
|
body: JSON.stringify(body) |
|
}); |
|
} |
|
|
|
let category = component.packname; |
|
if(component.category) { |
|
category += SEPARATOR + component.category; |
|
} |
|
if(category == '') { |
|
category = 'components'; |
|
} |
|
|
|
storeGroupNode(component_name, component); |
|
const config = new GroupNodeConfig(component_name, component); |
|
await config.registerType(category); |
|
} |
|
|
|
function restore_to_loaded_component(component_name) { |
|
if(rpack_map[component_name]) { |
|
let component = rpack_map[component_name]; |
|
storeGroupNode(component_name, component, false); |
|
const config = new GroupNodeConfig(component_name, component); |
|
config.registerType(component.category); |
|
} |
|
} |
|
|
|
|
|
let last_paste_timestamp = null; |
|
|
|
function versionCompare(v1, v2) { |
|
let ver1; |
|
let ver2; |
|
if(v1 && v1 != '') { |
|
ver1 = v1.split('.'); |
|
ver1[0] = parseInt(ver1[0]); |
|
ver1[1] = parseInt(ver1[1]); |
|
if(ver1.length == 2) |
|
ver1.push(0); |
|
else |
|
ver1[2] = parseInt(ver2[2]); |
|
} |
|
else { |
|
ver1 = [0,0,0]; |
|
} |
|
|
|
if(v2 && v2 != '') { |
|
ver2 = v2.split('.'); |
|
ver2[0] = parseInt(ver2[0]); |
|
ver2[1] = parseInt(ver2[1]); |
|
if(ver2.length == 2) |
|
ver2.push(0); |
|
else |
|
ver2[2] = parseInt(ver2[2]); |
|
} |
|
else { |
|
ver2 = [0,0,0]; |
|
} |
|
|
|
if(ver1[0] > ver2[0]) |
|
return -1; |
|
else if(ver1[0] < ver2[0]) |
|
return 1; |
|
|
|
if(ver1[1] > ver2[1]) |
|
return -1; |
|
else if(ver1[1] < ver2[1]) |
|
return 1; |
|
|
|
if(ver1[2] > ver2[2]) |
|
return -1; |
|
else if(ver1[2] < ver2[2]) |
|
return 1; |
|
|
|
return 0; |
|
} |
|
|
|
function checkVersion(name, component) { |
|
let msg = ''; |
|
if(rpack_map[name]) { |
|
let old_version = rpack_map[name].version; |
|
if(!old_version || old_version == '') { |
|
msg = ` '${name}' Upgrade (V0.0 -> V${component.version})`; |
|
} |
|
else { |
|
let c = versionCompare(old_version, component.version); |
|
if(c < 0) { |
|
msg = ` '${name}' Downgrade (V${old_version} -> V${component.version})`; |
|
} |
|
else if(c > 0) { |
|
msg = ` '${name}' Upgrade (V${old_version} -> V${component.version})`; |
|
} |
|
else { |
|
msg = ` '${name}' Same version (V${component.version})`; |
|
} |
|
} |
|
} |
|
else { |
|
msg = `'${name}' NEW (V${component.version})`; |
|
} |
|
|
|
return msg; |
|
} |
|
|
|
function handle_import_components(components) { |
|
let msg = 'Components:\n'; |
|
let cnt = 0; |
|
for(let name in components) { |
|
let component = components[name]; |
|
let v = checkVersion(name, component); |
|
|
|
if(cnt < 10) { |
|
msg += v + '\n'; |
|
} |
|
else if (cnt == 10) { |
|
msg += '...\n'; |
|
} |
|
else { |
|
|
|
} |
|
|
|
cnt++; |
|
} |
|
|
|
let last_name = null; |
|
msg += '\nWill you load components?\n'; |
|
if(confirm(msg)) { |
|
let mode = confirm('\nWill you save components?\n(cancel=load without save)'); |
|
|
|
for(let name in components) { |
|
let component = components[name]; |
|
import_component(name, component, mode); |
|
last_name = name; |
|
} |
|
|
|
if(mode) { |
|
show_message('Components are saved.'); |
|
} |
|
else { |
|
show_message('Components are loaded.'); |
|
} |
|
} |
|
|
|
if(cnt == 1 && last_name) { |
|
const node = LiteGraph.createNode(`workflow${SEPARATOR}${last_name}`); |
|
node.pos = [app.canvas.graph_mouse[0], app.canvas.graph_mouse[1]]; |
|
app.canvas.graph.add(node, false); |
|
} |
|
} |
|
|
|
function handlePaste(e) { |
|
let data = (e.clipboardData || window.clipboardData); |
|
const items = data.items; |
|
for(const item of items) { |
|
if(item.kind == 'string' && item.type == 'text/plain') { |
|
data = data.getData("text/plain"); |
|
try { |
|
let json_data = JSON.parse(data); |
|
if(json_data.kind == 'ComfyUI Components' && last_paste_timestamp != json_data.timestamp) { |
|
last_paste_timestamp = json_data.timestamp; |
|
handle_import_components(json_data.components); |
|
|
|
|
|
localStorage.removeItem("litegrapheditor_clipboard", null); |
|
} |
|
else { |
|
console.log('This components are already pasted: ignored'); |
|
} |
|
} |
|
catch { |
|
|
|
} |
|
} |
|
} |
|
} |
|
|
|
document.addEventListener("paste", handlePaste); |
|
|
|
|
|
export class ComponentBuilderDialog extends ComfyDialog { |
|
constructor() { |
|
super(); |
|
} |
|
|
|
clear() { |
|
while (this.element.children.length) { |
|
this.element.removeChild(this.element.children[0]); |
|
} |
|
} |
|
|
|
show() { |
|
this.invalidateControl(); |
|
|
|
this.element.style.display = "block"; |
|
this.element.style.zIndex = 10001; |
|
this.element.style.width = "500px"; |
|
this.element.style.height = "480px"; |
|
} |
|
|
|
invalidateControl() { |
|
this.clear(); |
|
|
|
let self = this; |
|
|
|
const close_button = $el("button", { id: "cm-close-button", type: "button", textContent: "Close", onclick: () => self.close() }); |
|
this.save_button = $el("button", |
|
{ id: "cm-save-button", type: "button", textContent: "Save", onclick: () => |
|
{ |
|
save_as_component(self.target_node, self.version_string.value.trim(), self.author.value.trim(), self.node_prefix.value.trim(), |
|
self.getNodeName(), self.getPackName(), self.category.value.trim()); |
|
} |
|
}); |
|
|
|
let default_nodename = getPureName(this.target_node).trim(); |
|
|
|
let groupNode = app.graph.extra.groupNodes[default_nodename]; |
|
let default_packname = groupNode.packname; |
|
if(!default_packname) { |
|
default_packname = ''; |
|
} |
|
|
|
let default_category = groupNode.category; |
|
if(!default_category) { |
|
default_category = ''; |
|
} |
|
|
|
this.default_ver = groupNode.version; |
|
if(!this.default_ver) { |
|
this.default_ver = '0.0'; |
|
} |
|
|
|
let default_author = groupNode.author; |
|
if(!default_author) { |
|
default_author = ''; |
|
} |
|
|
|
let delimiterIndex = default_nodename.indexOf('::'); |
|
let default_prefix = ""; |
|
if(delimiterIndex != -1) { |
|
default_prefix = default_nodename.substring(0, delimiterIndex); |
|
default_nodename = default_nodename.substring(delimiterIndex + 2); |
|
} |
|
|
|
if(!default_prefix) { |
|
this.save_button.disabled = true; |
|
} |
|
|
|
this.pack_list = this.createPackListCombo(); |
|
|
|
let version_string = this.createLabeledInput('input version (e.g. 1.0)', '*Version : ', this.default_ver); |
|
this.version_string = version_string[1]; |
|
this.version_string.disabled = true; |
|
|
|
let author = this.createLabeledInput('input author (e.g. Dr.Lt.Data)', 'Author : ', default_author); |
|
this.author = author[1]; |
|
|
|
let node_prefix = this.createLabeledInput('input node prefix (e.g. mypack)', '*Prefix : ', default_prefix); |
|
this.node_prefix = node_prefix[1]; |
|
|
|
let manual_nodename = this.createLabeledInput('input node name (e.g. MAKE_BASIC_PIPE)', 'Nodename : ', default_nodename); |
|
this.manual_nodename = manual_nodename[1]; |
|
|
|
let manual_packname = this.createLabeledInput('input pack name (e.g. mypack)', 'Packname : ', default_packname); |
|
this.manual_packname = manual_packname[1]; |
|
|
|
let category = this.createLabeledInput('input category (e.g. util/pipe)', 'Category : ', default_category); |
|
this.category = category[1]; |
|
|
|
this.node_label = this.createNodeLabel(); |
|
|
|
let author_mode = this.createAuthorModeCheck(); |
|
this.author_mode = author_mode[0]; |
|
|
|
const content = |
|
$el("div.comfy-modal-content", |
|
[ |
|
$el("tr.cm-title", {}, [ |
|
$el("font", {size:6, color:"white"}, [`ComfyUI-Manager: Component Builder`])] |
|
), |
|
$el("br", {}, []), |
|
$el("div.cm-menu-container", |
|
[ |
|
author_mode[0], |
|
author_mode[1], |
|
category[0], |
|
author[0], |
|
node_prefix[0], |
|
manual_nodename[0], |
|
manual_packname[0], |
|
version_string[0], |
|
this.pack_list, |
|
$el("br", {}, []), |
|
this.node_label |
|
]), |
|
|
|
$el("br", {}, []), |
|
this.save_button, |
|
close_button, |
|
] |
|
); |
|
|
|
content.style.width = '100%'; |
|
content.style.height = '100%'; |
|
|
|
this.element = $el("div.comfy-modal", { id:'cm-manager-dialog', parent: document.body }, [ content ]); |
|
} |
|
|
|
validateInput() { |
|
let msg = ""; |
|
|
|
if(!isValidVersionString(this.version_string.value)) { |
|
msg += 'Invalid version string: '+event.value+"\n"; |
|
} |
|
|
|
if(this.node_prefix.value.trim() == '') { |
|
msg += 'Node prefix cannot be empty\n'; |
|
} |
|
|
|
if(this.manual_nodename.value.trim() == '') { |
|
msg += 'Node name cannot be empty\n'; |
|
} |
|
|
|
if(msg != '') { |
|
|
|
} |
|
|
|
this.save_button.disabled = msg != ""; |
|
} |
|
|
|
getPackName() { |
|
if(this.pack_list.selectedIndex == 0) { |
|
return this.manual_packname.value.trim(); |
|
} |
|
|
|
return this.pack_list.value.trim(); |
|
} |
|
|
|
getNodeName() { |
|
if(this.manual_nodename.value.trim() != '') { |
|
return this.manual_nodename.value.trim(); |
|
} |
|
|
|
return getPureName(this.target_node); |
|
} |
|
|
|
createAuthorModeCheck() { |
|
let check = $el("input",{type:'checkbox', id:"author-mode"},[]) |
|
const check_label = $el("label",{for:"author-mode"},["Enable author mode"]); |
|
check_label.style.color = "var(--fg-color)"; |
|
check_label.style.cursor = "pointer"; |
|
check.checked = false; |
|
|
|
let self = this; |
|
check.onchange = () => { |
|
self.version_string.disabled = !check.checked; |
|
|
|
if(!check.checked) { |
|
self.version_string.value = self.default_ver; |
|
} |
|
else { |
|
alert('If you are not the author, it is not recommended to change the version, as it may cause component update issues.'); |
|
} |
|
}; |
|
|
|
return [check, check_label]; |
|
} |
|
|
|
createNodeLabel() { |
|
let label = $el('p'); |
|
label.className = 'cb-node-label'; |
|
if(this.target_node.comfyClass.includes('::')) |
|
label.textContent = getPureName(this.target_node); |
|
else |
|
label.textContent = " _::" + getPureName(this.target_node); |
|
return label; |
|
} |
|
|
|
createLabeledInput(placeholder, label, value) { |
|
let textbox = $el('input.cb-widget-input', {type:'text', placeholder:placeholder, value:value}, []); |
|
|
|
let self = this; |
|
textbox.onchange = () => { |
|
this.validateInput.call(self); |
|
this.node_label.textContent = this.node_prefix.value + "::" + this.manual_nodename.value; |
|
} |
|
let row = $el('span.cb-widget', {}, [ $el('span.cb-widget-input-label', label), textbox]); |
|
|
|
return [row, textbox]; |
|
} |
|
|
|
createPackListCombo() { |
|
let combo = document.createElement("select"); |
|
combo.className = "cb-widget"; |
|
let default_packname_option = { value: '##manual', text: 'Packname: Manual' }; |
|
|
|
combo.appendChild($el('option', default_packname_option, [])); |
|
for(let name in pack_map) { |
|
combo.appendChild($el('option', { value: name, text: 'Packname: '+ name }, [])); |
|
} |
|
|
|
let self = this; |
|
combo.onchange = function () { |
|
if(combo.selectedIndex == 0) { |
|
self.manual_packname.disabled = false; |
|
} |
|
else { |
|
self.manual_packname.disabled = true; |
|
} |
|
}; |
|
|
|
return combo; |
|
} |
|
} |
|
|
|
let orig_handleFile = app.handleFile; |
|
|
|
function handleFile(file) { |
|
if (file.name?.endsWith(".json") || file.name?.endsWith(".pack")) { |
|
const reader = new FileReader(); |
|
reader.onload = async () => { |
|
let is_component = false; |
|
const jsonContent = JSON.parse(reader.result); |
|
for(let name in jsonContent) { |
|
let cand = jsonContent[name]; |
|
is_component = cand.datetime && cand.version; |
|
break; |
|
} |
|
|
|
if(is_component) { |
|
handle_import_components(jsonContent); |
|
} |
|
else { |
|
orig_handleFile.call(app, file); |
|
} |
|
}; |
|
reader.readAsText(file); |
|
|
|
return; |
|
} |
|
|
|
orig_handleFile.call(app, file); |
|
} |
|
|
|
app.handleFile = handleFile; |
|
|
|
let current_component_policy = 'workflow'; |
|
try { |
|
api.fetchApi('/manager/component/policy') |
|
.then(response => response.text()) |
|
.then(data => { current_component_policy = data; }); |
|
} |
|
catch {} |
|
|
|
function getChangedVersion(groupNodes) { |
|
if(!Object.keys(pack_map).length || !groupNodes) |
|
return null; |
|
|
|
let res = {}; |
|
for(let component_name in groupNodes) { |
|
let data = groupNodes[component_name]; |
|
|
|
if(rpack_map[component_name]) { |
|
let v = versionCompare(data.version, rpack_map[component_name].version); |
|
res[component_name] = v; |
|
} |
|
} |
|
|
|
return res; |
|
} |
|
|
|
const loadGraphData = app.loadGraphData; |
|
app.loadGraphData = async function () { |
|
if(arguments.length == 0) |
|
return await loadGraphData.apply(this, arguments); |
|
|
|
let graphData = arguments[0]; |
|
let groupNodes = graphData.extra?.groupNodes; |
|
let res = getChangedVersion(groupNodes); |
|
|
|
if(res) { |
|
let target_components = null; |
|
switch(current_component_policy) { |
|
case 'higher': |
|
target_components = Object.keys(res).filter(key => res[key] == 1); |
|
break; |
|
|
|
case 'mine': |
|
target_components = Object.keys(res); |
|
break; |
|
|
|
default: |
|
|
|
} |
|
|
|
if(target_components) { |
|
for(let i in target_components) { |
|
let component_name = target_components[i]; |
|
let component = rpack_map[component_name]; |
|
if(component && graphData.extra?.groupNodes) { |
|
graphData.extra.groupNodes[component_name] = component; |
|
} |
|
} |
|
} |
|
} |
|
else { |
|
console.log('Empty components: policy ignored'); |
|
} |
|
|
|
arguments[0] = graphData; |
|
return await loadGraphData.apply(this, arguments); |
|
}; |
|
|
|
export function set_component_policy(v) { |
|
current_component_policy = v; |
|
} |
|
|
|
let graphToPrompt = app.graphToPrompt; |
|
app.graphToPrompt = async function () { |
|
let p = await graphToPrompt.call(app); |
|
try { |
|
let groupNodes = p.workflow.extra?.groupNodes; |
|
if(groupNodes) { |
|
p.workflow.extra = { ... p.workflow.extra}; |
|
|
|
|
|
let used_group_nodes = new Set(); |
|
for(let node of p.workflow.nodes) { |
|
if(node.type.startsWith(`workflow/`) || node.type.startsWith(`workflow${SEPARATOR}`)) { |
|
used_group_nodes.add(node.type.substring(9)); |
|
} |
|
} |
|
|
|
|
|
let new_groupNodes = {}; |
|
for (let key in p.workflow.extra.groupNodes) { |
|
if (used_group_nodes.has(key)) { |
|
new_groupNodes[key] = p.workflow.extra.groupNodes[key]; |
|
} |
|
} |
|
p.workflow.extra.groupNodes = new_groupNodes; |
|
} |
|
} |
|
catch(e) { |
|
console.log(`Failed to filtering group nodes: ${e}`); |
|
} |
|
|
|
return p; |
|
} |
|
|