Spaces:
Running
Running
import { app } from "../../scripts/app.js"; | |
import { api } from "../../scripts/api.js"; | |
import { ComfyDialog, $el } from "../../scripts/ui.js"; | |
import { GroupNodeConfig, GroupNodeHandler } from "./groupNode.js"; | |
// Adds the ability to save and add multiple nodes as a template | |
// To save: | |
// Select multiple nodes (ctrl + drag to select a region or ctrl+click individual nodes) | |
// Right click the canvas | |
// Save Node Template -> give it a name | |
// | |
// To add: | |
// Right click the canvas | |
// Node templates -> click the one to add | |
// | |
// To delete/rename: | |
// Right click the canvas | |
// Node templates -> Manage | |
// | |
// To rearrange: | |
// Open the manage dialog and Drag and drop elements using the "Name:" label as handle | |
const id = "Comfy.NodeTemplates"; | |
const file = "comfy.templates.json"; | |
class ManageTemplates extends ComfyDialog { | |
constructor() { | |
super(); | |
this.load().then((v) => { | |
this.templates = v; | |
}); | |
this.element.classList.add("comfy-manage-templates"); | |
this.draggedEl = null; | |
this.saveVisualCue = null; | |
this.emptyImg = new Image(); | |
this.emptyImg.src = ""; | |
this.importInput = $el("input", { | |
type: "file", | |
accept: ".json", | |
multiple: true, | |
style: { display: "none" }, | |
parent: document.body, | |
onchange: () => this.importAll(), | |
}); | |
} | |
createButtons() { | |
const btns = super.createButtons(); | |
btns[0].textContent = "Close"; | |
btns[0].onclick = (e) => { | |
clearTimeout(this.saveVisualCue); | |
this.close(); | |
}; | |
btns.unshift( | |
$el("button", { | |
type: "button", | |
textContent: "Export", | |
onclick: () => this.exportAll(), | |
}) | |
); | |
btns.unshift( | |
$el("button", { | |
type: "button", | |
textContent: "Import", | |
onclick: () => { | |
this.importInput.click(); | |
}, | |
}) | |
); | |
return btns; | |
} | |
async load() { | |
let templates = []; | |
if (app.storageLocation === "server") { | |
if (app.isNewUserSession) { | |
// New user so migrate existing templates | |
const json = localStorage.getItem(id); | |
if (json) { | |
templates = JSON.parse(json); | |
} | |
await api.storeUserData(file, json, { stringify: false }); | |
} else { | |
const res = await api.getUserData(file); | |
if (res.status === 200) { | |
try { | |
templates = await res.json(); | |
} catch (error) { | |
} | |
} else if (res.status !== 404) { | |
console.error(res.status + " " + res.statusText); | |
} | |
} | |
} else { | |
const json = localStorage.getItem(id); | |
if (json) { | |
templates = JSON.parse(json); | |
} | |
} | |
return templates ?? []; | |
} | |
async store() { | |
if(app.storageLocation === "server") { | |
const templates = JSON.stringify(this.templates, undefined, 4); | |
localStorage.setItem(id, templates); // Backwards compatibility | |
try { | |
await api.storeUserData(file, templates, { stringify: false }); | |
} catch (error) { | |
console.error(error); | |
alert(error.message); | |
} | |
} else { | |
localStorage.setItem(id, JSON.stringify(this.templates)); | |
} | |
} | |
async importAll() { | |
for (const file of this.importInput.files) { | |
if (file.type === "application/json" || file.name.endsWith(".json")) { | |
const reader = new FileReader(); | |
reader.onload = async () => { | |
const importFile = JSON.parse(reader.result); | |
if (importFile?.templates) { | |
for (const template of importFile.templates) { | |
if (template?.name && template?.data) { | |
this.templates.push(template); | |
} | |
} | |
await this.store(); | |
} | |
}; | |
await reader.readAsText(file); | |
} | |
} | |
this.importInput.value = null; | |
this.close(); | |
} | |
exportAll() { | |
if (this.templates.length == 0) { | |
alert("No templates to export."); | |
return; | |
} | |
const json = JSON.stringify({ templates: this.templates }, null, 2); // convert the data to a JSON string | |
const blob = new Blob([json], { type: "application/json" }); | |
const url = URL.createObjectURL(blob); | |
const a = $el("a", { | |
href: url, | |
download: "node_templates.json", | |
style: { display: "none" }, | |
parent: document.body, | |
}); | |
a.click(); | |
setTimeout(function () { | |
a.remove(); | |
window.URL.revokeObjectURL(url); | |
}, 0); | |
} | |
show() { | |
// Show list of template names + delete button | |
super.show( | |
$el( | |
"div", | |
{}, | |
this.templates.flatMap((t,i) => { | |
let nameInput; | |
return [ | |
$el( | |
"div", | |
{ | |
dataset: { id: i }, | |
className: "tempateManagerRow", | |
style: { | |
display: "grid", | |
gridTemplateColumns: "1fr auto", | |
border: "1px dashed transparent", | |
gap: "5px", | |
backgroundColor: "var(--comfy-menu-bg)" | |
}, | |
ondragstart: (e) => { | |
this.draggedEl = e.currentTarget; | |
e.currentTarget.style.opacity = "0.6"; | |
e.currentTarget.style.border = "1px dashed yellow"; | |
e.dataTransfer.effectAllowed = 'move'; | |
e.dataTransfer.setDragImage(this.emptyImg, 0, 0); | |
}, | |
ondragend: (e) => { | |
e.target.style.opacity = "1"; | |
e.currentTarget.style.border = "1px dashed transparent"; | |
e.currentTarget.removeAttribute("draggable"); | |
// rearrange the elements | |
this.element.querySelectorAll('.tempateManagerRow').forEach((el,i) => { | |
var prev_i = el.dataset.id; | |
if ( el == this.draggedEl && prev_i != i ) { | |
this.templates.splice(i, 0, this.templates.splice(prev_i, 1)[0]); | |
} | |
el.dataset.id = i; | |
}); | |
this.store(); | |
}, | |
ondragover: (e) => { | |
e.preventDefault(); | |
if ( e.currentTarget == this.draggedEl ) | |
return; | |
let rect = e.currentTarget.getBoundingClientRect(); | |
if (e.clientY > rect.top + rect.height / 2) { | |
e.currentTarget.parentNode.insertBefore(this.draggedEl, e.currentTarget.nextSibling); | |
} else { | |
e.currentTarget.parentNode.insertBefore(this.draggedEl, e.currentTarget); | |
} | |
} | |
}, | |
[ | |
$el( | |
"label", | |
{ | |
textContent: "Name: ", | |
style: { | |
cursor: "grab", | |
}, | |
onmousedown: (e) => { | |
// enable dragging only from the label | |
if (e.target.localName == 'label') | |
e.currentTarget.parentNode.draggable = 'true'; | |
} | |
}, | |
[ | |
$el("input", { | |
value: t.name, | |
dataset: { name: t.name }, | |
style: { | |
transitionProperty: 'background-color', | |
transitionDuration: '0s', | |
}, | |
onchange: (e) => { | |
clearTimeout(this.saveVisualCue); | |
var el = e.target; | |
var row = el.parentNode.parentNode; | |
this.templates[row.dataset.id].name = el.value.trim() || 'untitled'; | |
this.store(); | |
el.style.backgroundColor = 'rgb(40, 95, 40)'; | |
el.style.transitionDuration = '0s'; | |
this.saveVisualCue = setTimeout(function () { | |
el.style.transitionDuration = '.7s'; | |
el.style.backgroundColor = 'var(--comfy-input-bg)'; | |
}, 15); | |
}, | |
onkeypress: (e) => { | |
var el = e.target; | |
clearTimeout(this.saveVisualCue); | |
el.style.transitionDuration = '0s'; | |
el.style.backgroundColor = 'var(--comfy-input-bg)'; | |
}, | |
$: (el) => (nameInput = el), | |
}) | |
] | |
), | |
$el( | |
"div", | |
{}, | |
[ | |
$el("button", { | |
textContent: "Export", | |
style: { | |
fontSize: "12px", | |
fontWeight: "normal", | |
}, | |
onclick: (e) => { | |
const json = JSON.stringify({templates: [t]}, null, 2); // convert the data to a JSON string | |
const blob = new Blob([json], {type: "application/json"}); | |
const url = URL.createObjectURL(blob); | |
const a = $el("a", { | |
href: url, | |
download: (nameInput.value || t.name) + ".json", | |
style: {display: "none"}, | |
parent: document.body, | |
}); | |
a.click(); | |
setTimeout(function () { | |
a.remove(); | |
window.URL.revokeObjectURL(url); | |
}, 0); | |
}, | |
}), | |
$el("button", { | |
textContent: "Delete", | |
style: { | |
fontSize: "12px", | |
color: "red", | |
fontWeight: "normal", | |
}, | |
onclick: (e) => { | |
const item = e.target.parentNode.parentNode; | |
item.parentNode.removeChild(item); | |
this.templates.splice(item.dataset.id*1, 1); | |
this.store(); | |
// update the rows index, setTimeout ensures that the list is updated | |
var that = this; | |
setTimeout(function (){ | |
that.element.querySelectorAll('.tempateManagerRow').forEach((el,i) => { | |
el.dataset.id = i; | |
}); | |
}, 0); | |
}, | |
}), | |
] | |
), | |
] | |
) | |
]; | |
}) | |
) | |
); | |
} | |
} | |
app.registerExtension({ | |
name: id, | |
setup() { | |
const manage = new ManageTemplates(); | |
const clipboardAction = async (cb) => { | |
// We use the clipboard functions but dont want to overwrite the current user clipboard | |
// Restore it after we've run our callback | |
const old = localStorage.getItem("litegrapheditor_clipboard"); | |
await cb(); | |
localStorage.setItem("litegrapheditor_clipboard", old); | |
}; | |
const orig = LGraphCanvas.prototype.getCanvasMenuOptions; | |
LGraphCanvas.prototype.getCanvasMenuOptions = function () { | |
const options = orig.apply(this, arguments); | |
options.push(null); | |
options.push({ | |
content: `Save Selected as Template`, | |
disabled: !Object.keys(app.canvas.selected_nodes || {}).length, | |
callback: () => { | |
const name = prompt("Enter name"); | |
if (!name?.trim()) return; | |
clipboardAction(() => { | |
app.canvas.copyToClipboard(); | |
let data = localStorage.getItem("litegrapheditor_clipboard"); | |
data = JSON.parse(data); | |
const nodeIds = Object.keys(app.canvas.selected_nodes); | |
for (let i = 0; i < nodeIds.length; i++) { | |
const node = app.graph.getNodeById(nodeIds[i]); | |
const nodeData = node?.constructor.nodeData; | |
let groupData = GroupNodeHandler.getGroupData(node); | |
if (groupData) { | |
groupData = groupData.nodeData; | |
if (!data.groupNodes) { | |
data.groupNodes = {}; | |
} | |
data.groupNodes[nodeData.name] = groupData; | |
data.nodes[i].type = nodeData.name; | |
} | |
} | |
manage.templates.push({ | |
name, | |
data: JSON.stringify(data), | |
}); | |
manage.store(); | |
}); | |
}, | |
}); | |
// Map each template to a menu item | |
const subItems = manage.templates.map((t) => { | |
return { | |
content: t.name, | |
callback: () => { | |
clipboardAction(async () => { | |
const data = JSON.parse(t.data); | |
await GroupNodeConfig.registerFromWorkflow(data.groupNodes, {}); | |
localStorage.setItem("litegrapheditor_clipboard", t.data); | |
app.canvas.pasteFromClipboard(); | |
}); | |
}, | |
}; | |
}); | |
subItems.push(null, { | |
content: "Manage", | |
callback: () => manage.show(), | |
}); | |
options.push({ | |
content: "Node Templates", | |
submenu: { | |
options: subItems, | |
}, | |
}); | |
return options; | |
}; | |
}, | |
}); | |