Spaces:
Running
on
Zero
Running
on
Zero
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; | |
}, | |
}); | |
} | |
} | |
}, | |
}); | |