Spaces:
Build error
Build error
| 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; | |
| }, | |
| }); | |
| } | |
| } | |
| }, | |
| }); | |