|
import { ComfyApp, app } from "../../scripts/app.js";
|
|
|
|
export function register_concat_conditionings_with_multiplier_node(nodeType, nodeData, app) {
|
|
if (nodeData.name === 'ConcatConditioningsWithMultiplier //Inspire') {
|
|
var input_name = "conditioning";
|
|
|
|
const onConnectionsChange = nodeType.prototype.onConnectionsChange;
|
|
let this_handler = async function (type, index, connected, link_info) {
|
|
let last_state = this.state_change_handling;
|
|
try {
|
|
this.state_change_handling = true;
|
|
if(!link_info || link_info.type != 'CONDITIONING')
|
|
return;
|
|
|
|
let self = this;
|
|
|
|
function get_input_count(prefix, linked_only) {
|
|
let cnt = 0;
|
|
for(let i in self.inputs) {
|
|
if(linked_only && !self.inputs[i].link)
|
|
continue;
|
|
|
|
if(self.inputs[i].name.startsWith(prefix))
|
|
cnt+=1;
|
|
}
|
|
|
|
return cnt;
|
|
}
|
|
|
|
function get_widget_count(prefix) {
|
|
let cnt = 0;
|
|
for(let i in self.widgets) {
|
|
if(self.widgets[i].name.startsWith(prefix))
|
|
cnt+=1;
|
|
}
|
|
|
|
return cnt;
|
|
}
|
|
|
|
function get_unconnected() {
|
|
let unconnected = [];
|
|
for(let i in self.inputs) {
|
|
let input = self.inputs[i];
|
|
if(input.name.startsWith('conditioning')) {
|
|
if(input.link == undefined)
|
|
unconnected.push(i);
|
|
}
|
|
}
|
|
|
|
return unconnected;
|
|
}
|
|
|
|
let unconnected = get_unconnected();
|
|
|
|
function renames() {
|
|
let con_i = 1;
|
|
|
|
let rename_map = {};
|
|
|
|
for(let i in self.inputs) {
|
|
let input = self.inputs[i];
|
|
if(input.name.startsWith('conditioning')) {
|
|
let orig_i = Number(input.name.substring(12));
|
|
if(orig_i != con_i) {
|
|
rename_map[orig_i] = con_i;
|
|
input.name = 'conditioning'+con_i;
|
|
}
|
|
con_i++;
|
|
}
|
|
}
|
|
|
|
|
|
for(let i in self.inputs) {
|
|
let input = self.inputs[i];
|
|
if(input.name.startsWith('multiplier')) {
|
|
let orig_i = Number(input.name.substring(10));
|
|
if(rename_map[orig_i]) {
|
|
input.name = 'multiplier'+rename_map[orig_i];
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
for(let i in self.widgets) {
|
|
let w = self.widgets[i];
|
|
if(w.name.startsWith('multiplier')) {
|
|
let orig_i = Number(w.name.substring(10));
|
|
if(rename_map[orig_i]) {
|
|
w.name = 'multiplier'+rename_map[orig_i];
|
|
}
|
|
}
|
|
}
|
|
|
|
return con_i;
|
|
}
|
|
|
|
function remove_multiplier_link(i, link_id) {
|
|
let link = app.graph.links[link_id];
|
|
const node = app.graph.getNodeById(link.origin_id);
|
|
let x = node.outputs[link.origin_slot].links.findIndex((w) => w == link_id);
|
|
node.outputs[link.origin_slot].links.splice(x, 1);
|
|
self.disconnectInput(i);
|
|
app.graph.links.splice(link_id, 1);
|
|
}
|
|
|
|
async function remove_target_multiplier(target_name) {
|
|
|
|
for(let i in self.inputs) {
|
|
let input = self.inputs[i];
|
|
if(input.name.startsWith(target_name)) {
|
|
if(input.link) {
|
|
remove_multiplier_link(i, input.link);
|
|
}
|
|
await self.removeInput(i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
const widget_index = self.widgets.findIndex((w) => w.name == target_name);
|
|
self.widgets.splice(widget_index, 1);
|
|
}
|
|
|
|
async function remove_garbage() {
|
|
let unconnected = get_unconnected();
|
|
|
|
|
|
while(unconnected.length > 0) {
|
|
let last_one = unconnected.reverse()[0];
|
|
self.removeInput(last_one);
|
|
unconnected = get_unconnected();
|
|
}
|
|
|
|
|
|
let conds = new Set();
|
|
let muls = new Set();
|
|
for(let i in self.inputs) {
|
|
let input = self.inputs[i];
|
|
if(input.link && input.name.startsWith('conditioning')) {
|
|
let index = Number(input.name.substring(12));
|
|
conds.add(index);
|
|
}
|
|
else if(input.name.startsWith('multiplier')) {
|
|
let index = Number(input.name.substring(10));
|
|
muls.add(index);
|
|
}
|
|
}
|
|
for(let i in self.widgets) {
|
|
let index = Number(self.widgets[i].name.substring(10));
|
|
muls.add(index);
|
|
}
|
|
|
|
let dangling_muls = [...muls].filter(x => !conds.has(x));
|
|
while(dangling_muls.length > 0) {
|
|
let remove_target = dangling_muls.pop();
|
|
let target_name = `multiplier${remove_target}`;
|
|
await remove_target_multiplier(target_name);
|
|
}
|
|
}
|
|
|
|
async function ensure_multipliers() {
|
|
if(self.ensuring_multipliers) {
|
|
return;
|
|
}
|
|
try {
|
|
self.ensuring_multipliers = true;
|
|
|
|
let ncon = get_input_count('conditioning', true);
|
|
let nmul = get_input_count('multiplier', false) + get_widget_count('multiplier');
|
|
|
|
if(ncon == 0 && nmul == 0)
|
|
ncon = 1;
|
|
|
|
for(let i = nmul+1; i<=ncon; i++) {
|
|
let config = { min: 0, max: 10, step: 0.1, round: 0.01, precision: 2 };
|
|
|
|
|
|
let widget = await self.addWidget("number", `multiplier${i}`, 1.0, function (v) {
|
|
if (config.round) {
|
|
self.value = Math.round(v/config.round)*config.round;
|
|
} else {
|
|
self.value = v;
|
|
}
|
|
}, config);
|
|
}
|
|
}
|
|
finally{
|
|
self.ensuring_multipliers = null;
|
|
}
|
|
}
|
|
|
|
async function recover_multipliers() {
|
|
if(self.recover_multipliers) {
|
|
return;
|
|
}
|
|
try {
|
|
self.recover_multipliers = true;
|
|
for(let i = 1; i<self.widgets_values.length; i++) {
|
|
let config = { min: 0, max: 10, step: 0.1, round: 0.01, precision: 2 };
|
|
|
|
|
|
let widget = await self.addWidget("number", `multiplier${i+1}`, 1.0, function (v) {
|
|
if (config.round) {
|
|
self.value = Math.round(v/config.round)*config.round;
|
|
} else {
|
|
self.value = v;
|
|
}
|
|
}, config);
|
|
}
|
|
}
|
|
finally{
|
|
self.recover_multipliers = null;
|
|
}
|
|
}
|
|
|
|
async function ensure_inputs() {
|
|
if(get_unconnected() == 0) {
|
|
let con_i = renames();
|
|
self.addInput(`conditioning${con_i}`, self.outputs[0].type);
|
|
}
|
|
}
|
|
|
|
const stackTrace = new Error().stack;
|
|
if(!stackTrace.includes('loadGraphData') && !stackTrace.includes('pasteFromClipboard')) {
|
|
await remove_garbage();
|
|
await ensure_inputs();
|
|
}
|
|
|
|
if(!stackTrace.includes('loadGraphData')) {
|
|
await ensure_multipliers();
|
|
}
|
|
else {
|
|
await recover_multipliers();
|
|
}
|
|
|
|
await this.setSize( this.computeSize() );
|
|
}
|
|
finally {
|
|
this.state_change_handling = last_state;
|
|
}
|
|
}
|
|
|
|
nodeType.prototype.onConnectionsChange = this_handler;
|
|
}
|
|
}
|
|
|
|
function ensure_splitter_outputs(node, output_name, value, type) {
|
|
if(node.outputs.length != (value + 1)) {
|
|
while(node.outputs.length != (value + 1)) {
|
|
if(node.outputs.length > value + 1) {
|
|
node.removeOutput(node.outputs.length-1);
|
|
}
|
|
else {
|
|
node.addOutput(`output${node.outputs.length+1}`, type);
|
|
}
|
|
}
|
|
|
|
for(let i in node.outputs) {
|
|
let output = node.outputs[i];
|
|
output.name = `${output_name} ${parseInt(i)+1}`;
|
|
}
|
|
|
|
if(node.outputs[0].label == type || node.outputs[0].label == 'remained')
|
|
delete node.outputs[0].label;
|
|
|
|
|
|
let last_output = node.outputs[node.outputs.length-1];
|
|
last_output.name = 'remained';
|
|
}
|
|
}
|
|
|
|
export function register_splitter(node, app) {
|
|
if(node.comfyClass === 'ImageBatchSplitter //Inspire' || node.comfyClass === 'LatentBatchSplitter //Inspire') {
|
|
let split_count = node.widgets[0];
|
|
|
|
let output_name = 'output';
|
|
let output_type = "*";
|
|
|
|
if(node.comfyClass === 'ImageBatchSplitter //Inspire') {
|
|
output_name = 'image';
|
|
output_type = "IMAGE";
|
|
}
|
|
else if(node.comfyClass === 'LatentBatchSplitter //Inspire') {
|
|
output_name = 'latent';
|
|
output_type = "LATENT";
|
|
}
|
|
|
|
ensure_splitter_outputs(node, output_name, split_count.value, output_type);
|
|
|
|
Object.defineProperty(split_count, "value", {
|
|
set: async function(value) {
|
|
if(value < 0 || value > 50)
|
|
return;
|
|
|
|
ensure_splitter_outputs(node, output_name, value, output_type);
|
|
},
|
|
get: function() {
|
|
return node.outputs.length - 1;
|
|
}
|
|
});
|
|
}
|
|
} |