3v324v23's picture
lfs
1e3b872
raw
history blame
7.52 kB
import { app } from "../../../scripts/app.js";
// Allows you to manage preset tags for e.g. common negative prompt
// Also performs replacements on any text field e.g. allowing you to use preset text in CLIP Text encode fields
let replaceRegex;
const id = "pysssss.PresetText.Presets";
const MISSING = Symbol();
const getPresets = () => {
let items;
try {
items = JSON.parse(localStorage.getItem(id));
} catch (error) {}
if (!items || !items.length) {
items = [{ name: "default negative", value: "worst quality" }];
}
return items;
};
let presets = getPresets();
app.registerExtension({
name: "pysssss.PresetText",
setup() {
app.ui.settings.addSetting({
id: "pysssss.PresetText.ReplacementRegex",
name: "๐Ÿ Preset Text Replacement Regex",
type: "text",
defaultValue: "(?:^|[^\\w])(?<replace>@(?<id>[\\w-]+))",
tooltip:
"The regex should return two named capture groups: id (the name of the preset text to use), replace (the matched text to replace)",
attrs: {
style: {
fontFamily: "monospace",
},
},
onChange(value) {
if (!value) {
replaceRegex = null;
return;
}
try {
replaceRegex = new RegExp(value, "g");
} catch (error) {
alert("Error creating regex for preset text replacement, no replacements will be performed.");
replaceRegex = null;
}
},
});
const drawNodeWidgets = LGraphCanvas.prototype.drawNodeWidgets
LGraphCanvas.prototype.drawNodeWidgets = function(node) {
const c = LiteGraph.WIDGET_BGCOLOR;
try {
if(node[MISSING]) {
LiteGraph.WIDGET_BGCOLOR = "red"
}
return drawNodeWidgets.apply(this, arguments);
} finally {
LiteGraph.WIDGET_BGCOLOR = c;
}
}
},
registerCustomNodes() {
class PresetTextNode {
constructor() {
this.isVirtualNode = true;
this.serialize_widgets = true;
this.addOutput("text", "STRING");
const widget = this.addWidget("combo", "value", presets[0].name, () => {}, {
values: presets.map((p) => p.name),
});
this.addWidget("button", "Manage", "Manage", () => {
const container = document.createElement("div");
Object.assign(container.style, {
display: "grid",
gridTemplateColumns: "1fr 1fr",
gap: "10px",
});
const addNew = document.createElement("button");
addNew.textContent = "Add New";
addNew.classList.add("pysssss-presettext-addnew");
Object.assign(addNew.style, {
fontSize: "13px",
gridColumn: "1 / 3",
color: "dodgerblue",
width: "auto",
textAlign: "center",
});
addNew.onclick = () => {
addRow({ name: "", value: "" });
};
container.append(addNew);
function addRow(p) {
const name = document.createElement("input");
const nameLbl = document.createElement("label");
name.value = p.name;
nameLbl.textContent = "Name:";
nameLbl.append(name);
const value = document.createElement("input");
const valueLbl = document.createElement("label");
value.value = p.value;
valueLbl.textContent = "Value:";
valueLbl.append(value);
addNew.before(nameLbl, valueLbl);
}
for (const p of presets) {
addRow(p);
}
const help = document.createElement("span");
help.textContent = "To remove a preset set the name or value to blank";
help.style.gridColumn = "1 / 3";
container.append(help);
dialog.show("");
dialog.textElement.append(container);
});
const dialog = new app.ui.dialog.constructor();
dialog.element.classList.add("comfy-settings");
const closeButton = dialog.element.querySelector("button");
closeButton.textContent = "CANCEL";
const saveButton = document.createElement("button");
saveButton.textContent = "SAVE";
saveButton.onclick = function () {
const inputs = dialog.element.querySelectorAll("input");
const p = [];
for (let i = 0; i < inputs.length; i += 2) {
const n = inputs[i];
const v = inputs[i + 1];
if (!n.value.trim() || !v.value.trim()) {
continue;
}
p.push({ name: n.value, value: v.value });
}
widget.options.values = p.map((p) => p.name);
if (!widget.options.values.includes(widget.value)) {
widget.value = widget.options.values[0];
}
presets = p;
localStorage.setItem(id, JSON.stringify(presets));
dialog.close();
};
closeButton.before(saveButton);
this.applyToGraph = function (workflow) {
// For each output link copy our value over the original widget value
if (this.outputs[0].links && this.outputs[0].links.length) {
for (const l of this.outputs[0].links) {
const link_info = app.graph.links[l];
const outNode = app.graph.getNodeById(link_info.target_id);
const outIn = outNode && outNode.inputs && outNode.inputs[link_info.target_slot];
if (outIn.widget) {
const w = outNode.widgets.find((w) => w.name === outIn.widget.name);
if (!w) continue;
const preset = presets.find((p) => p.name === widget.value);
if (!preset) {
this[MISSING] = true;
app.graph.setDirtyCanvas(true, true);
const msg = `Preset text '${widget.value}' not found. Please fix this and queue again.`;
throw new Error(msg);
}
delete this[MISSING];
w.value = preset.value;
}
}
}
};
}
}
LiteGraph.registerNodeType(
"PresetText|pysssss",
Object.assign(PresetTextNode, {
title: "Preset Text ๐Ÿ",
})
);
PresetTextNode.category = "utils";
},
nodeCreated(node) {
if (node.widgets) {
// Locate dynamic prompt text widgets
const widgets = node.widgets.filter((n) => n.type === "customtext" || n.type === "text");
for (const widget of widgets) {
const callbacks = [
() => {
let prompt = widget.value;
if (replaceRegex && typeof prompt.replace !== 'undefined') {
prompt = prompt.replace(replaceRegex, (match, p1, p2, index, text, groups) => {
if (!groups.replace || !groups.id) return match; // No match, bad regex?
const preset = presets.find((p) => p.name.replaceAll(/\s/g, "-") === groups.id);
if (!preset) return match; // Invalid name
const pos = match.indexOf(groups.replace);
return match.substring(0, pos) + preset.value;
});
}
return prompt;
},
];
let inheritedSerializeValue = widget.serializeValue || null;
let called = false;
const serializeValue = async (workflowNode, widgetIndex) => {
const origWidgetValue = widget.value;
if (called) return origWidgetValue;
called = true;
let allCallbacks = [...callbacks];
if (inheritedSerializeValue) {
allCallbacks.push(inheritedSerializeValue)
}
let valueIsUndefined = false;
for (const cb of allCallbacks) {
let value = await cb(workflowNode, widgetIndex);
// Need to check the callback return value before it is set on widget.value as it coerces it to a string (even for undefined)
if (value === undefined) valueIsUndefined = true;
widget.value = value;
}
const prompt = valueIsUndefined ? undefined : widget.value;
widget.value = origWidgetValue;
called = false;
return prompt;
};
Object.defineProperty(widget, "serializeValue", {
get() {
return serializeValue;
},
set(cb) {
inheritedSerializeValue = cb;
},
});
}
}
},
});