| | import { $el, ComfyDialog } from "../../scripts/ui.js";
|
| | import { DraggableList } from "../../scripts/ui/draggableList.js";
|
| | import { addStylesheet } from "../../scripts/utils.js";
|
| | import { GroupNodeConfig, GroupNodeHandler } from "./groupNode.js";
|
| |
|
| | addStylesheet(import.meta.url);
|
| |
|
| | const ORDER = Symbol();
|
| |
|
| | function merge(target, source) {
|
| | if (typeof target === "object" && typeof source === "object") {
|
| | for (const key in source) {
|
| | const sv = source[key];
|
| | if (typeof sv === "object") {
|
| | let tv = target[key];
|
| | if (!tv) tv = target[key] = {};
|
| | merge(tv, source[key]);
|
| | } else {
|
| | target[key] = sv;
|
| | }
|
| | }
|
| | }
|
| |
|
| | return target;
|
| | }
|
| |
|
| | export class ManageGroupDialog extends ComfyDialog {
|
| |
|
| | tabs = {};
|
| |
|
| | selectedNodeIndex;
|
| |
|
| | selectedTab = "Inputs";
|
| |
|
| | selectedGroup;
|
| |
|
| |
|
| | modifications = {};
|
| |
|
| | get selectedNodeInnerIndex() {
|
| | return +this.nodeItems[this.selectedNodeIndex].dataset.nodeindex;
|
| | }
|
| |
|
| | constructor(app) {
|
| | super();
|
| | this.app = app;
|
| | this.element = $el("dialog.comfy-group-manage", {
|
| | parent: document.body,
|
| | });
|
| | }
|
| |
|
| | changeTab(tab) {
|
| | this.tabs[this.selectedTab].tab.classList.remove("active");
|
| | this.tabs[this.selectedTab].page.classList.remove("active");
|
| | this.tabs[tab].tab.classList.add("active");
|
| | this.tabs[tab].page.classList.add("active");
|
| | this.selectedTab = tab;
|
| | }
|
| |
|
| | changeNode(index, force) {
|
| | if (!force && this.selectedNodeIndex === index) return;
|
| |
|
| | if (this.selectedNodeIndex != null) {
|
| | this.nodeItems[this.selectedNodeIndex].classList.remove("selected");
|
| | }
|
| | this.nodeItems[index].classList.add("selected");
|
| | this.selectedNodeIndex = index;
|
| |
|
| | if (!this.buildInputsPage() && this.selectedTab === "Inputs") {
|
| | this.changeTab("Widgets");
|
| | }
|
| | if (!this.buildWidgetsPage() && this.selectedTab === "Widgets") {
|
| | this.changeTab("Outputs");
|
| | }
|
| | if (!this.buildOutputsPage() && this.selectedTab === "Outputs") {
|
| | this.changeTab("Inputs");
|
| | }
|
| |
|
| | this.changeTab(this.selectedTab);
|
| | }
|
| |
|
| | getGroupData() {
|
| | this.groupNodeType = LiteGraph.registered_node_types["workflow/" + this.selectedGroup];
|
| | this.groupNodeDef = this.groupNodeType.nodeData;
|
| | this.groupData = GroupNodeHandler.getGroupData(this.groupNodeType);
|
| | }
|
| |
|
| | changeGroup(group, reset = true) {
|
| | this.selectedGroup = group;
|
| | this.getGroupData();
|
| |
|
| | const nodes = this.groupData.nodeData.nodes;
|
| | this.nodeItems = nodes.map((n, i) =>
|
| | $el(
|
| | "li.draggable-item",
|
| | {
|
| | dataset: {
|
| | nodeindex: n.index + "",
|
| | },
|
| | onclick: () => {
|
| | this.changeNode(i);
|
| | },
|
| | },
|
| | [
|
| | $el("span.drag-handle"),
|
| | $el(
|
| | "div",
|
| | {
|
| | textContent: n.title ?? n.type,
|
| | },
|
| | n.title
|
| | ? $el("span", {
|
| | textContent: n.type,
|
| | })
|
| | : []
|
| | ),
|
| | ]
|
| | )
|
| | );
|
| |
|
| | this.innerNodesList.replaceChildren(...this.nodeItems);
|
| |
|
| | if (reset) {
|
| | this.selectedNodeIndex = null;
|
| | this.changeNode(0);
|
| | } else {
|
| | const items = this.draggable.getAllItems();
|
| | let index = items.findIndex(item => item.classList.contains("selected"));
|
| | if(index === -1) index = this.selectedNodeIndex;
|
| | this.changeNode(index, true);
|
| | }
|
| |
|
| | const ordered = [...nodes];
|
| | this.draggable?.dispose();
|
| | this.draggable = new DraggableList(this.innerNodesList, "li");
|
| | this.draggable.addEventListener("dragend", ({ detail: { oldPosition, newPosition } }) => {
|
| | if (oldPosition === newPosition) return;
|
| | ordered.splice(newPosition, 0, ordered.splice(oldPosition, 1)[0]);
|
| | for (let i = 0; i < ordered.length; i++) {
|
| | this.storeModification({ nodeIndex: ordered[i].index, section: ORDER, prop: "order", value: i });
|
| | }
|
| | });
|
| | }
|
| |
|
| | storeModification({ nodeIndex, section, prop, value }) {
|
| | const groupMod = (this.modifications[this.selectedGroup] ??= {});
|
| | const nodesMod = (groupMod.nodes ??= {});
|
| | const nodeMod = (nodesMod[nodeIndex ?? this.selectedNodeInnerIndex] ??= {});
|
| | const typeMod = (nodeMod[section] ??= {});
|
| | if (typeof value === "object") {
|
| | const objMod = (typeMod[prop] ??= {});
|
| | Object.assign(objMod, value);
|
| | } else {
|
| | typeMod[prop] = value;
|
| | }
|
| | }
|
| |
|
| | getEditElement(section, prop, value, placeholder, checked, checkable = true) {
|
| | if (value === placeholder) value = "";
|
| |
|
| | const mods = this.modifications[this.selectedGroup]?.nodes?.[this.selectedNodeInnerIndex]?.[section]?.[prop];
|
| | if (mods) {
|
| | if (mods.name != null) {
|
| | value = mods.name;
|
| | }
|
| | if (mods.visible != null) {
|
| | checked = mods.visible;
|
| | }
|
| | }
|
| |
|
| | return $el("div", [
|
| | $el("input", {
|
| | value,
|
| | placeholder,
|
| | type: "text",
|
| | onchange: (e) => {
|
| | this.storeModification({ section, prop, value: { name: e.target.value } });
|
| | },
|
| | }),
|
| | $el("label", { textContent: "Visible" }, [
|
| | $el("input", {
|
| | type: "checkbox",
|
| | checked,
|
| | disabled: !checkable,
|
| | onchange: (e) => {
|
| | this.storeModification({ section, prop, value: { visible: !!e.target.checked } });
|
| | },
|
| | }),
|
| | ]),
|
| | ]);
|
| | }
|
| |
|
| | buildWidgetsPage() {
|
| | const widgets = this.groupData.oldToNewWidgetMap[this.selectedNodeInnerIndex];
|
| | const items = Object.keys(widgets ?? {});
|
| | const type = app.graph.extra.groupNodes[this.selectedGroup];
|
| | const config = type.config?.[this.selectedNodeInnerIndex]?.input;
|
| | this.widgetsPage.replaceChildren(
|
| | ...items.map((oldName) => {
|
| | return this.getEditElement("input", oldName, widgets[oldName], oldName, config?.[oldName]?.visible !== false);
|
| | })
|
| | );
|
| | return !!items.length;
|
| | }
|
| |
|
| | buildInputsPage() {
|
| | const inputs = this.groupData.nodeInputs[this.selectedNodeInnerIndex];
|
| | const items = Object.keys(inputs ?? {});
|
| | const type = app.graph.extra.groupNodes[this.selectedGroup];
|
| | const config = type.config?.[this.selectedNodeInnerIndex]?.input;
|
| | this.inputsPage.replaceChildren(
|
| | ...items
|
| | .map((oldName) => {
|
| | let value = inputs[oldName];
|
| | if (!value) {
|
| | return;
|
| | }
|
| |
|
| | return this.getEditElement("input", oldName, value, oldName, config?.[oldName]?.visible !== false);
|
| | })
|
| | .filter(Boolean)
|
| | );
|
| | return !!items.length;
|
| | }
|
| |
|
| | buildOutputsPage() {
|
| | const nodes = this.groupData.nodeData.nodes;
|
| | const innerNodeDef = this.groupData.getNodeDef(nodes[this.selectedNodeInnerIndex]);
|
| | const outputs = innerNodeDef?.output ?? [];
|
| | const groupOutputs = this.groupData.oldToNewOutputMap[this.selectedNodeInnerIndex];
|
| |
|
| | const type = app.graph.extra.groupNodes[this.selectedGroup];
|
| | const config = type.config?.[this.selectedNodeInnerIndex]?.output;
|
| | const node = this.groupData.nodeData.nodes[this.selectedNodeInnerIndex];
|
| | const checkable = node.type !== "PrimitiveNode";
|
| | this.outputsPage.replaceChildren(
|
| | ...outputs
|
| | .map((type, slot) => {
|
| | const groupOutputIndex = groupOutputs?.[slot];
|
| | const oldName = innerNodeDef.output_name?.[slot] ?? type;
|
| | let value = config?.[slot]?.name;
|
| | const visible = config?.[slot]?.visible || groupOutputIndex != null;
|
| | if (!value || value === oldName) {
|
| | value = "";
|
| | }
|
| | return this.getEditElement("output", slot, value, oldName, visible, checkable);
|
| | })
|
| | .filter(Boolean)
|
| | );
|
| | return !!outputs.length;
|
| | }
|
| |
|
| | show(type) {
|
| | const groupNodes = Object.keys(app.graph.extra?.groupNodes ?? {}).sort((a, b) => a.localeCompare(b));
|
| |
|
| | this.innerNodesList = $el("ul.comfy-group-manage-list-items");
|
| | this.widgetsPage = $el("section.comfy-group-manage-node-page");
|
| | this.inputsPage = $el("section.comfy-group-manage-node-page");
|
| | this.outputsPage = $el("section.comfy-group-manage-node-page");
|
| | const pages = $el("div", [this.widgetsPage, this.inputsPage, this.outputsPage]);
|
| |
|
| | this.tabs = [
|
| | ["Inputs", this.inputsPage],
|
| | ["Widgets", this.widgetsPage],
|
| | ["Outputs", this.outputsPage],
|
| | ].reduce((p, [name, page]) => {
|
| | p[name] = {
|
| | tab: $el("a", {
|
| | onclick: () => {
|
| | this.changeTab(name);
|
| | },
|
| | textContent: name,
|
| | }),
|
| | page,
|
| | };
|
| | return p;
|
| | }, {});
|
| |
|
| | const outer = $el("div.comfy-group-manage-outer", [
|
| | $el("header", [
|
| | $el("h2", "Group Nodes"),
|
| | $el(
|
| | "select",
|
| | {
|
| | onchange: (e) => {
|
| | this.changeGroup(e.target.value);
|
| | },
|
| | },
|
| | groupNodes.map((g) =>
|
| | $el("option", {
|
| | textContent: g,
|
| | selected: "workflow/" + g === type,
|
| | value: g,
|
| | })
|
| | )
|
| | ),
|
| | ]),
|
| | $el("main", [
|
| | $el("section.comfy-group-manage-list", this.innerNodesList),
|
| | $el("section.comfy-group-manage-node", [
|
| | $el(
|
| | "header",
|
| | Object.values(this.tabs).map((t) => t.tab)
|
| | ),
|
| | pages,
|
| | ]),
|
| | ]),
|
| | $el("footer", [
|
| | $el(
|
| | "button.comfy-btn",
|
| | {
|
| | onclick: (e) => {
|
| | const node = app.graph._nodes.find((n) => n.type === "workflow/" + this.selectedGroup);
|
| | if (node) {
|
| | alert("This group node is in use in the current workflow, please first remove these.");
|
| | return;
|
| | }
|
| | if (confirm(`Are you sure you want to remove the node: "${this.selectedGroup}"`)) {
|
| | delete app.graph.extra.groupNodes[this.selectedGroup];
|
| | LiteGraph.unregisterNodeType("workflow/" + this.selectedGroup);
|
| | }
|
| | this.show();
|
| | },
|
| | },
|
| | "Delete Group Node"
|
| | ),
|
| | $el(
|
| | "button.comfy-btn",
|
| | {
|
| | onclick: async () => {
|
| | let nodesByType;
|
| | let recreateNodes = [];
|
| | const types = {};
|
| | for (const g in this.modifications) {
|
| | const type = app.graph.extra.groupNodes[g];
|
| | let config = (type.config ??= {});
|
| |
|
| | let nodeMods = this.modifications[g]?.nodes;
|
| | if (nodeMods) {
|
| | const keys = Object.keys(nodeMods);
|
| | if (nodeMods[keys[0]][ORDER]) {
|
| |
|
| | const orderedNodes = [];
|
| | const orderedMods = {};
|
| | const orderedConfig = {};
|
| |
|
| | for (const n of keys) {
|
| | const order = nodeMods[n][ORDER].order;
|
| | orderedNodes[order] = type.nodes[+n];
|
| | orderedMods[order] = nodeMods[n];
|
| | orderedNodes[order].index = order;
|
| | }
|
| |
|
| |
|
| | for (const l of type.links) {
|
| | if (l[0] != null) l[0] = type.nodes[l[0]].index;
|
| | if (l[2] != null) l[2] = type.nodes[l[2]].index;
|
| | }
|
| |
|
| |
|
| | if (type.external) {
|
| | for (const ext of type.external) {
|
| | ext[0] = type.nodes[ext[0]];
|
| | }
|
| | }
|
| |
|
| |
|
| | for (const id of keys) {
|
| | if (config[id]) {
|
| | orderedConfig[type.nodes[id].index] = config[id];
|
| | }
|
| | delete config[id];
|
| | }
|
| |
|
| | type.nodes = orderedNodes;
|
| | nodeMods = orderedMods;
|
| | type.config = config = orderedConfig;
|
| | }
|
| |
|
| | merge(config, nodeMods);
|
| | }
|
| |
|
| | types[g] = type;
|
| |
|
| | if (!nodesByType) {
|
| | nodesByType = app.graph._nodes.reduce((p, n) => {
|
| | p[n.type] ??= [];
|
| | p[n.type].push(n);
|
| | return p;
|
| | }, {});
|
| | }
|
| |
|
| | const nodes = nodesByType["workflow/" + g];
|
| | if (nodes) recreateNodes.push(...nodes);
|
| | }
|
| |
|
| | await GroupNodeConfig.registerFromWorkflow(types, {});
|
| |
|
| | for (const node of recreateNodes) {
|
| | node.recreate();
|
| | }
|
| |
|
| | this.modifications = {};
|
| | this.app.graph.setDirtyCanvas(true, true);
|
| | this.changeGroup(this.selectedGroup, false);
|
| | },
|
| | },
|
| | "Save"
|
| | ),
|
| | $el("button.comfy-btn", { onclick: () => this.element.close() }, "Close"),
|
| | ]),
|
| | ]);
|
| |
|
| | this.element.replaceChildren(outer);
|
| | this.changeGroup(type ? groupNodes.find((g) => "workflow/" + g === type) : groupNodes[0]);
|
| | this.element.showModal();
|
| |
|
| | this.element.addEventListener("close", () => {
|
| | this.draggable?.dispose();
|
| | });
|
| | }
|
| | } |