Merge pull request #28 from tusharbhutt/dev

Try #2 to get font change script added
This commit is contained in:
tusharbhutt
2025-07-19 16:28:37 -06:00
committed by GitHub
11 changed files with 575 additions and 6 deletions

View File

@@ -2,7 +2,7 @@
Some basic custom nodes for the ComfyUI user interface for Stable Diffusion. Features:
+ **True batch multiprompting capability for ComfyUI**
+ **True batch multi-prompting capability for ComfyUI**
+ An image saver for images and JSON files to base folder, custom folders for one, or custom folders for both. Also allows for Python timestamping
+ Switches for text and numbers
+ Random prompt selectors
@@ -14,6 +14,14 @@ When using the [ComfyUI](https://github.com/comfyanonymous/ComfyUI) interface fo
Rightly or wrongly, I was teaching myself a bit of Python back in 2023 to get some nodes up and running to do what I'd like, and I am starting to do that again. Yes, I am using ChatGPT, Copilot, Claude and others, and yes, I am a glutton for punishment. There are no promises that these nodes will work for you or that I will maintain them. Feel free to do with them as you wish, according to the license model.
***
**UPDATE: JUL 18, 2025**
**Version 1.3 introduces the Endless 🌊✨ Fontifier, a little button on your taskbar that allows you to dynamically change fonts and sizes.**
+ No need to dive into CSS to change text size
+ Allows changes to the title areas, connector text, widgets, and more
+ Adjust the higher of the title bar and other areas too, to accommodate the new font size
***
**UPDATE: JUL 8, 2025**
@@ -64,9 +72,35 @@ I am not a programmer, nor do I care to be. I have a fulltime job that eats up
If you have issues, ask me **nicely** for help. Your tone matters; I'm too old and tired to pay attention to people who think I blew up their machines, and if how I react to you if you are difficult bothers you, some self-reflection is in order on your part. You are not "forthright" or "honest" or "direct", you're merely an ass if you think badgering people is justifiable to get what you want. The world has too many assholes, don't make me think you're another one.
***
## Button List
### Endless 🌊✨ Fontifier
I always found it odd that in the early days of ComfyUI, you could not change the font size for various node elements. Sure you could manually go into the CSS styling in a user file, but that is not user friendly. Later versions have allowed you to change the widget text size, but that's it. Yes, you can zoom in, but... now you've lost your larger view of the workflow. If you have a 4K monitor and old eyes, too bad so sad for you. This javacsript places a button on your task bar called "Endless 🌊✨ Fontifier". Clicking it shows the dialog box below:
![fontifierbox](./img/fontifierbox.png)
Drag the box aronud by clicking and holding the title. To cancel, you can simply click outside the dialog box or press the escape key. With this dialog box, you can do the following:
+ Globally change the font size for all text elements
+ Change the fonts themselves
+ Instead of a global change, select various elements to resize
+ Adjust the higher of the title bar or connectors and other input areas
Once you make your changes, you can preview them and then choose to apply or cancel. Changed your mind? Load the box again and press the reset key.
![fontifiernode](./img/fontifiernode.png)
**NOTE: There is a few bugs still where the changes don't respect the preview or reset buttons, instead they are applied immediately. I'll fix these issues soon.**
## Node List
### Batch Multiprompt Node for SD, SDXL, FLUX, and FLUX Kontext
### Batch Multiprompt Node for SD, SDXL, and FLUX
![fluxbatch](./img/batchprompts.png)

View File

@@ -1,4 +1,8 @@
July 7/25, V 1.2.4: You can now use the batch prompt node with Flux Kontext Dev. The process works the same way as the other nodes, except here the prompts are used to make changes to the image(s). You cannot iterate within the prompt set (e.g, set up a list of prompts for a sequence of changes), it is designed to allow you to process multiple scenarios at once.
July 19/26, V1.3: INtroducing the Endless Fontifier, a javascript file that adds allows the user to change font sizes and fonts for various text elements on the ComfyUI interface.
July 8/25, V1.2.5: Fixed bug in Image Saver that forced a connection for prompts. That is now optional
July 7/25, V1.2.4: You can now use the batch prompt node with Flux Kontext Dev. The process works the same way as the other nodes, except here the prompts are used to make changes to the image(s). You cannot iterate within the prompt set (e.g, set up a list of prompts for a sequence of changes), it is designed to allow you to process multiple scenarios at once.
July 6/25, V1.2.3: Corrected the JSON files that were being exported. They will now load the workflow when dragged back to the UI. As a bonus, if the PNGINfo is also selected, the JSON will remove that, lowering the size of the file.

BIN
img/fontifierbox.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 45 KiB

BIN
img/fontifiernode.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 26 KiB

BIN
img/imagescorer.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 199 KiB

BIN
img/saliency.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 242 KiB

BIN
img/scoreinput.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 MiB

BIN
img/scoreinput2.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 819 KiB

BIN
img/structural.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 288 KiB

View File

@@ -1,13 +1,13 @@
[project]
name = "endless-nodes"
version = "1.2.5"
description = "Modular, prompt-aware nodes for ComfyUI—analyze images, rank prompts, embed with CLIP, batch with control, and explore aesthetic/structural depth. 🌊✨"
description = "A small set of nodes I created for myself. Features multiple simultaneous prompts in batches, an image saver with ability to have JSON saved to separate folder, image analysis nodes, switches for text and numbers, and more."
version = "1.3.0"
license = { file = "LICENSE" }
readme = "README.md"
dependencies = ""
[project.urls]
Repository = "https://github.com/tusharbhutt/Endless-Nodes"
# Used by Comfy Registry https://comfyregistry.org
[tool.comfy]
PublisherId = "tusharbhutt"

531
web/endless_fontifier.js Normal file
View File

@@ -0,0 +1,531 @@
// ComfyUI Endless 🌊✨ Fontifier - Improved Version
(function() {
'use strict';
// Store original values for reset functionality
const originalValues = {
NODE_TEXT_SIZE: 14,
NODE_SUBTEXT_SIZE: 12,
NODE_TITLE_HEIGHT: 30,
DEFAULT_GROUP_FONT: 24,
NODE_FONT: 'Arial',
NODE_SLOT_HEIGHT: 20,
NODE_WIDGET_HEIGHT: 20
};
// Current values (will be updated as user changes them)
let currentValues = { ...originalValues };
// Get ComfyUI theme colors
function getComfyUIColors() {
const computedStyle = getComputedStyle(document.documentElement);
return {
background: computedStyle.getPropertyValue('--comfy-menu-bg') || '#353535',
backgroundSecondary: computedStyle.getPropertyValue('--comfy-input-bg') || '#222',
border: computedStyle.getPropertyValue('--border-color') || '#999',
text: computedStyle.getPropertyValue('--input-text') || '#ddd',
textSecondary: computedStyle.getPropertyValue('--descrip-text') || '#999',
accent: computedStyle.getPropertyValue('--comfy-menu-bg') || '#0f0f0f'
};
}
function makeDraggable(dialog) {
const header = dialog.querySelector('h2');
if (!header) return;
let offsetX = 0, offsetY = 0, isDown = false;
header.style.cursor = 'move';
header.style.userSelect = 'none';
header.onmousedown = (e) => {
e.preventDefault();
isDown = true;
// Get the actual position of the dialog
const rect = dialog.getBoundingClientRect();
offsetX = e.clientX - rect.left;
offsetY = e.clientY - rect.top;
const onMouseMove = (e) => {
if (!isDown) return;
e.preventDefault();
dialog.style.left = `${e.clientX - offsetX}px`;
dialog.style.top = `${e.clientY - offsetY}px`;
dialog.style.transform = 'none'; // Remove the centering transform
};
const onMouseUp = () => {
isDown = false;
document.removeEventListener('mousemove', onMouseMove);
document.removeEventListener('mouseup', onMouseUp);
};
document.addEventListener('mousemove', onMouseMove);
document.addEventListener('mouseup', onMouseUp);
};
}
function createFontifierDialog() {
// Remove existing dialog if present
const existingDialog = document.getElementById('fontifier-dialog');
if (existingDialog) {
existingDialog.remove();
}
const colors = getComfyUIColors();
// Create dialog container
const dialog = document.createElement('div');
dialog.id = 'fontifier-dialog';
dialog.className = 'comfyui-dialog';
dialog.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: ${colors.background};
border: 1px solid ${colors.border};
border-radius: 8px;
padding: 20px;
z-index: 10000;
width: 520px;
max-height: 80vh;
overflow-y: auto;
font-family: Arial, sans-serif;
box-shadow: 0 4px 12px rgba(0,0,0,0.3);
color: ${colors.text};
`;
// Create backdrop
const backdrop = document.createElement('div');
backdrop.className = 'comfyui-backdrop';
backdrop.style.cssText = `
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(0,0,0,0.5);
z-index: 9999;
`;
backdrop.onclick = () => {
backdrop.remove();
dialog.remove();
};
dialog.innerHTML = `
<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 12px; border-bottom: 1px solid ${colors.border}; padding-bottom: 15px;">
<h2 style="color: ${colors.text}; margin: 0; font-size: 16px;">🌊✨ Endless Fontifier</h2>
<button id="close-dialog" style="background: ${colors.backgroundSecondary}; border: 1px solid ${colors.border}; color: ${colors.text}; padding: 6px 12px; border-radius: 4px; cursor: pointer;">✕</button>
</div>
<div style="margin-bottom: 12px; padding: 12px; background: ${colors.backgroundSecondary}; border-radius: 6px; border: 1px solid ${colors.border};">
<h3 style="color: ${colors.text}; margin: 0 0 10px 0; font-size: 16px;">Global Scale</h3>
<div style="display: flex; align-items: center; gap: 12px;">
<label style="color: ${colors.textSecondary}; min-width: 80px; font-size: 12px;">Scale All:</label>
<input type="range" id="global-scale" min="0.5" max="3" step="0.1" value="1" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="global-scale-num" min="0.5" max="3" step="0.1" value="1" style="width: 70px; padding: 6px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 12px; padding: 12px; background: ${colors.backgroundSecondary}; border-radius: 6px; border: 1px solid ${colors.border};">
<h3 style="color: ${colors.text}; margin: 0 0 12px 0; font-size: 16px;">Font Family</h3>
<select id="font-family" style="width: 100%; padding: 8px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
<option value="Arial">Arial</option>
<option value="Helvetica">Helvetica</option>
<option value="Times New Roman">Times New Roman</option>
<option value="Courier New">Courier New</option>
<option value="Verdana">Verdana</option>
<option value="Georgia">Georgia</option>
<option value="Comic Sans MS">Comic Sans MS</option>
<option value="Impact">Impact</option>
<option value="Trebuchet MS">Trebuchet MS</option>
<option value="Tahoma">Tahoma</option>
</select>
</div>
<div style="margin-bottom: 12px; padding: 12px; background: ${colors.backgroundSecondary}; border-radius: 6px; border: 1px solid ${colors.border};">
<h3 style="color: ${colors.text}; margin: 0 0 12px 0; font-size: 16px;">Text Element Sizes</h3>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Node Title Text</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">The main title text at the top of each node (e.g., "KSampler", "VAE Decode")</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="node-text-size" min="8" max="32" value="${currentValues.NODE_TEXT_SIZE}" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="node-text-size-num" min="8" max="32" value="${currentValues.NODE_TEXT_SIZE}" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Widget Labels & Values</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">Text inside nodes: parameter names and values (e.g., "steps: 20", "cfg: 8.0")</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="node-subtext-size" min="6" max="24" value="${currentValues.NODE_SUBTEXT_SIZE}" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="node-subtext-size-num" min="6" max="24" value="${currentValues.NODE_SUBTEXT_SIZE}" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Widget Text Input Size</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">Font size for text inside input boxes, dropdowns, and textareas in nodes.</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="widget-text-size" min="8" max="24" value="12" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="widget-text-size-num" min="8" max="24" value="12" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Node Title Area Height</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">Height of the colored title bar area at the top of nodes</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="title-height" min="20" max="60" value="${currentValues.NODE_TITLE_HEIGHT}" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="title-height-num" min="20" max="60" value="${currentValues.NODE_TITLE_HEIGHT}" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Connection Slot Height</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">Height of input/output connection points on node sides</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="slot-height" min="12" max="40" value="${currentValues.NODE_SLOT_HEIGHT}" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="slot-height-num" min="12" max="40" value="${currentValues.NODE_SLOT_HEIGHT}" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
<div style="margin-bottom: 10px;">
<label style="color: ${colors.text}; display: block; margin-bottom: 4px; font-size: 12px; font-weight: bold;">Group Label Size</label>
<div style="color: ${colors.textSecondary}; font-size: 10px; margin-bottom: 5px;">Text size for node group labels (when nodes are grouped together)</div>
<div style="display: flex; align-items: center; gap: 10px;">
<input type="range" id="group-font-size" min="12" max="48" value="${currentValues.DEFAULT_GROUP_FONT}" style="flex: 1; accent-color: ${colors.accent};">
<input type="number" id="group-font-size-num" min="12" max="48" value="${currentValues.DEFAULT_GROUP_FONT}" style="width: 60px; padding: 5px; background: ${colors.background}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; font-size: 12px;">
</div>
</div>
</div>
<div style="display: flex; gap: 10px; justify-content: center; padding-top: 15px; border-top: 1px solid ${colors.border};">
<button id="reset-btn" style="padding: 8px 16px; background: ${colors.backgroundSecondary}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; cursor: pointer; font-size: 12px; transition: border-width 0.2s ease;">Reset</button>
<button id="preview-btn" style="padding: 8px 16px; background: ${colors.backgroundSecondary}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; cursor: pointer; font-size: 12px; transition: border-width 0.2s ease;">Preview</button>
<button id="apply-btn" style="padding: 8px 16px; background: ${colors.backgroundSecondary}; border: 1px solid ${colors.border}; color: ${colors.text}; border-radius: 4px; cursor: pointer; font-size: 12px; transition: border-width 0.2s ease; background-image: linear-gradient(rgba(128, 255, 128, 0.08), rgba(128, 255, 128, 0.08));">Apply & Close</button>
<button id="cancel-btn" style="padding: 8px 16px; background: ${colors.backgroundSecondary}; border: 1px solid ${colors.border}; color: ${colors.textSecondary}; border-radius: 4px; cursor: pointer; font-size: 12px; transition: border-width 0.2s ease; background-image: linear-gradient(rgba(255, 128, 128, 0.08), rgba(255, 128, 128, 0.08));">Cancel</button>
</div>
`;
document.body.appendChild(backdrop);
document.body.appendChild(dialog);
// ESC key handler
document.addEventListener('keydown', function escHandler(e) {
if (e.key === 'Escape') {
backdrop.remove();
dialog.remove();
document.removeEventListener('keydown', escHandler);
}
});
// Set up event handlers
setupDialogHandlers(dialog, backdrop);
}
function setupDialogHandlers(dialog, backdrop) {
// call drag function
makeDraggable(dialog);
// Sync sliders with number inputs
const elements = [
'global-scale',
'node-text-size',
'node-subtext-size',
'title-height',
'slot-height',
'group-font-size',
'widget-text-size'
];
elements.forEach(id => {
const slider = dialog.querySelector(`#${id}`);
const numberInput = dialog.querySelector(`#${id}-num`);
slider.oninput = () => {
numberInput.value = slider.value;
// Update global scale number input properly
if (id === 'global-scale') {
const globalScaleNum = dialog.querySelector('#global-scale-num');
globalScaleNum.value = slider.value;
}
};
numberInput.oninput = () => {
slider.value = numberInput.value;
};
});
// Global scale handler
const globalScale = dialog.querySelector('#global-scale');
const globalScaleNum = dialog.querySelector('#global-scale-num');
function updateGlobalScale() {
const scale = parseFloat(globalScale.value);
globalScaleNum.value = scale; // Fix: Update the number input
// Update all individual controls
const updates = [
['node-text-size', originalValues.NODE_TEXT_SIZE],
['node-subtext-size', originalValues.NODE_SUBTEXT_SIZE],
['title-height', originalValues.NODE_TITLE_HEIGHT],
['slot-height', originalValues.NODE_SLOT_HEIGHT],
['group-font-size', originalValues.DEFAULT_GROUP_FONT]
];
updates.forEach(([id, originalValue]) => {
const newValue = Math.round(originalValue * scale);
dialog.querySelector(`#${id}`).value = newValue;
dialog.querySelector(`#${id}-num`).value = newValue;
});
}
globalScale.oninput = updateGlobalScale;
globalScaleNum.oninput = () => {
globalScale.value = globalScaleNum.value;
updateGlobalScale();
};
// Button handlers
dialog.querySelector('#close-dialog').onclick = () => {
backdrop.remove();
dialog.remove();
};
dialog.querySelector('#reset-btn').onclick = () => {
dialog.querySelector('#global-scale').value = 1;
dialog.querySelector('#global-scale-num').value = 1;
dialog.querySelector('#node-text-size').value = originalValues.NODE_TEXT_SIZE;
dialog.querySelector('#node-text-size-num').value = originalValues.NODE_TEXT_SIZE;
dialog.querySelector('#node-subtext-size').value = originalValues.NODE_SUBTEXT_SIZE;
dialog.querySelector('#node-subtext-size-num').value = originalValues.NODE_SUBTEXT_SIZE;
dialog.querySelector('#title-height').value = originalValues.NODE_TITLE_HEIGHT;
dialog.querySelector('#title-height-num').value = originalValues.NODE_TITLE_HEIGHT;
dialog.querySelector('#slot-height').value = originalValues.NODE_SLOT_HEIGHT;
dialog.querySelector('#slot-height-num').value = originalValues.NODE_SLOT_HEIGHT;
dialog.querySelector('#group-font-size').value = originalValues.DEFAULT_GROUP_FONT;
dialog.querySelector('#group-font-size-num').value = originalValues.DEFAULT_GROUP_FONT;
dialog.querySelector('#font-family').value = 'Arial';
};
dialog.querySelector('#preview-btn').onclick = () => applyChanges(dialog, false);
dialog.querySelector('#apply-btn').onclick = () => {
applyChanges(dialog, true);
backdrop.remove();
dialog.remove();
};
dialog.querySelector('#cancel-btn').onclick = () => {
backdrop.remove();
dialog.remove();
};
// Add hover effects to buttons
const buttons = dialog.querySelectorAll('button');
buttons.forEach(button => {
button.style.boxSizing = 'border-box';
button.style.minWidth = button.offsetWidth + 'px'; // Lock the width
button.addEventListener('mouseenter', () => {
button.style.borderWidth = '2px';
button.style.padding = '7px 15px';
});
button.addEventListener('mouseleave', () => {
button.style.borderWidth = '1px';
button.style.padding = '8px 16px';
});
});
}
function applyChanges(dialog, permanent = false) {
const newValues = {
NODE_TEXT_SIZE: parseInt(dialog.querySelector('#node-text-size').value),
NODE_SUBTEXT_SIZE: parseInt(dialog.querySelector('#node-subtext-size').value),
NODE_TITLE_HEIGHT: parseInt(dialog.querySelector('#title-height').value),
NODE_SLOT_HEIGHT: parseInt(dialog.querySelector('#slot-height').value),
DEFAULT_GROUP_FONT: parseInt(dialog.querySelector('#group-font-size').value),
FONT_FAMILY: dialog.querySelector('#font-family').value
};
if (typeof LiteGraph !== 'undefined') {
LiteGraph.NODE_TEXT_SIZE = newValues.NODE_TEXT_SIZE;
LiteGraph.NODE_SUBTEXT_SIZE = newValues.NODE_SUBTEXT_SIZE;
LiteGraph.NODE_TITLE_HEIGHT = newValues.NODE_TITLE_HEIGHT;
LiteGraph.NODE_SLOT_HEIGHT = newValues.NODE_SLOT_HEIGHT;
LiteGraph.NODE_WIDGET_HEIGHT = newValues.NODE_SLOT_HEIGHT;
LiteGraph.DEFAULT_GROUP_FONT = newValues.DEFAULT_GROUP_FONT;
LiteGraph.DEFAULT_GROUP_FONT_SIZE = newValues.DEFAULT_GROUP_FONT;
LiteGraph.NODE_FONT = newValues.FONT_FAMILY;
LiteGraph.DEFAULT_FONT = newValues.FONT_FAMILY;
LiteGraph.GROUP_FONT = newValues.FONT_FAMILY;
console.log('🌊✨ Fontifier applied:', newValues);
if (typeof app !== 'undefined' && app.canvas) {
app.canvas.setDirty(true, true);
if (app.canvas.draw) {
setTimeout(() => app.canvas.draw(true, true), 100);
}
}
const canvases = document.querySelectorAll('canvas');
canvases.forEach(canvas => {
if (canvas.getContext) {
const ctx = canvas.getContext('2d');
const originalWidth = canvas.width;
canvas.width = originalWidth + 1;
canvas.width = originalWidth;
}
});
}
// Apply widget font size to CSS, this is DOM-only
const widgetTextSize = parseInt(dialog.querySelector('#widget-text-size').value);
let styleTag = document.getElementById('fontifier-widget-text-style');
if (!styleTag) {
styleTag = document.createElement('style');
styleTag.id = 'fontifier-widget-text-style';
document.head.appendChild(styleTag);
}
styleTag.textContent = `
canvas ~ * .widget input, canvas ~ * .widget select, canvas ~ * .widget textarea,
canvas ~ * .comfy-multiline-input, canvas ~ * .comfy-input,
canvas ~ * input.comfy-multiline-input, canvas ~ * textarea.comfy-multiline-input,
canvas ~ * [class*="comfy-input"], canvas ~ * [class*="comfy-multiline"],
canvas ~ * .comfyui-widget input, canvas ~ * .comfyui-widget select, canvas ~ * .comfyui-widget textarea,
canvas ~ * [class*="widget"] input, canvas ~ * [class*="widget"] select, canvas ~ * [class*="widget"] textarea,
canvas ~ * .litegraph input, canvas ~ * .litegraph select, canvas ~ * .litegraph textarea,
.litegraph input, .litegraph select, .litegraph textarea {
font-size: ${widgetTextSize}px !important;
font-family: ${newValues.FONT_FAMILY} !important;
}
/* Exclude the fontifier dialog itself */
#fontifier-dialog input, #fontifier-dialog select, #fontifier-dialog textarea {
font-size: 14px !important;
font-family: Arial !important;
}
`;
if (permanent) {
currentValues = { ...newValues };
console.log('🌊✨ Fontifier changes applied permanently (until page refresh)');
}
}
function findToolbar() {
// Method 1: Look for ComfyUI specific toolbar classes
let toolbar = document.querySelector('.comfyui-menu, .comfy-menu, [class*="menu"], [class*="toolbar"]');
// Method 2: Look for button groups
if (!toolbar) {
const buttonGroups = document.querySelectorAll('[class*="button-group"], [class*="btn-group"], .comfyui-button-group');
toolbar = Array.from(buttonGroups).find(group =>
group.querySelectorAll('button').length > 0
);
}
// Method 3: Look for any container with multiple buttons
if (!toolbar) {
const allElements = document.querySelectorAll('*');
toolbar = Array.from(allElements).find(el => {
const buttons = el.querySelectorAll('button');
return buttons.length >= 2 && buttons.length <= 10; // Reasonable toolbar size
});
}
// Method 4: Fallback to the original Share button method
if (!toolbar) {
toolbar = Array.from(document.querySelectorAll(".comfyui-button-group")).find(div =>
Array.from(div.querySelectorAll("button")).some(btn => btn.title === "Share")
);
}
return toolbar;
}
function injectFontifierButton() {
const toolbar = findToolbar();
if (toolbar && !document.getElementById("endless-fontifier-button")) {
const colors = getComfyUIColors();
const btn = document.createElement("button");
btn.id = "endless-fontifier-button";
btn.textContent = "🌊✨ Fontifier";
btn.className = "comfyui-button";
// Function to update button colors
function updateButtonColors() {
const currentColors = getComfyUIColors();
btn.style.cssText = `
margin-left: 8px;
background: ${currentColors.backgroundSecondary};
border: 1px solid ${currentColors.border};
color: ${currentColors.text};
padding: 6px 12px;
border-radius: 4px;
cursor: pointer;
font-size: 14px;
transition: all 0.2s ease;
`;
btn.onmouseover = () => {
const hoverColors = getComfyUIColors();
btn.style.background = hoverColors.background;
btn.style.borderColor = hoverColors.text;
};
btn.onmouseout = () => {
const outColors = getComfyUIColors();
btn.style.background = outColors.backgroundSecondary;
btn.style.borderColor = outColors.border;
};
}
// Initial colors
updateButtonColors();
// Watch for theme changes
const observer = new MutationObserver(updateButtonColors);
observer.observe(document.documentElement, {
attributes: true,
attributeFilter: ['class', 'style']
});
btn.onclick = createFontifierDialog;
toolbar.appendChild(btn);
console.log("✅ 🌊✨ Endless Fontifier button injected successfully!");
return true;
}
return false;
}
// Try to inject immediately
if (!injectFontifierButton()) {
// If immediate injection fails, use observer
const observer = new MutationObserver(() => {
if (injectFontifierButton()) {
observer.disconnect();
}
});
observer.observe(document.body, { childList: true, subtree: true });
// Timeout after 30 seconds to avoid infinite observation
setTimeout(() => {
observer.disconnect();
if (!document.getElementById("endless-fontifier-button")) {
console.warn("⚠️ Could not find suitable toolbar for Fontifier button");
}
}, 30000);
}
})();