Spaces:
Running
on
Zero
Running
on
Zero
/** | |
* Adds a named stylesheet to the document with an optional ability to replace an existing one. | |
* | |
* @param {string} name - The unique name (ID) of the stylesheet. | |
* @param {string} css - The CSS rules as a string. | |
* @param {boolean} [force=false] - Whether to replace the existing stylesheet if it exists. | |
* @returns {void} | |
*/ | |
export function addNamedStyleSheet(name, css, force = false) { | |
const existingStyleSheet = document.getElementById(name) | |
if (existingStyleSheet && !force) { | |
console.debug( | |
`Stylesheet with name "${name}" already exists. Skipping addition.`, | |
) | |
return | |
} | |
if (existingStyleSheet && force) { | |
console.debug(`Stylesheet with name "${name}" exists. Replacing...`) | |
existingStyleSheet.remove() | |
} | |
const styleElement = document.createElement('style') | |
styleElement.id = name | |
styleElement.type = 'text/css' | |
styleElement.appendChild(document.createTextNode(css)) | |
document.head.appendChild(styleElement) | |
console.debug(`Stylesheet with name "${name}" added.`) | |
} | |
export const ensureMTBStyles = () => { | |
const S = { | |
fg: 'var(--fg-color)', | |
bgi: 'var(--comfy-input-bg)', | |
bgm: 'var(--comfy-menu-bg)', | |
border: 'var(--comfy-border)', | |
borderHover: 'var(--comfy-border-hover)', | |
box: 'var(--comfy-box)', | |
accent: 'var(--p-button-text-primary-color)', | |
} | |
const common = ` | |
.mtb_sidebar { | |
display: flex; | |
flex-direction: column; | |
background: ${S.bgm}; | |
} | |
.mtb_img_grid { | |
display: flex; | |
flex-wrap: wrap; | |
overflow: scroll; | |
gap: 1em; | |
align-items: center; | |
justify-content: center; | |
height: 100%; | |
width: 100%; | |
} | |
.mtb_tools { | |
display: flex; | |
flex-direction: row; | |
align-items: center; | |
justify-content: space-between; | |
width: 100%; | |
} | |
` | |
const inputs = ` | |
/* SELECT */ | |
.mtb_select { | |
appearance: none; | |
display: grid; | |
grid-template-areas: "select"; | |
padding: 10px; | |
background-color: ${S.bgi}; | |
border: none; | |
border-radius: 5px; | |
font-size: 14px; | |
color: ${S.fg}; | |
cursor: pointer; | |
width: 100%; | |
} | |
@supports (-moz-appearance:none) { | |
.mtb_select{ | |
grid-area: select; | |
background: ${S.bgi} url('') right center no-repeat !important; | |
background-position: calc(100% - 5px) center !important; | |
-moz-appearance:none !important; | |
} | |
/* styling the dropdown arrow for browsers that support it */ | |
.mtb_select:after { | |
content: ""; | |
width: 0.8em; | |
height: 0.5em; | |
background-color: ${S.fg}; | |
clip-path: polygon(100% 0%, 0 0%, 50% 100%); | |
} | |
.mtb_select:focus { | |
outline: none; | |
border-color: #0056b3; | |
} | |
.mtb_select > option { | |
padding: 10px; | |
background-color: ${S.bgi}; | |
border:none; | |
color: ${S.fg}; | |
} | |
.mtb_select > option:hover { | |
background-color: red; | |
color: ${S.fg}; | |
} | |
/* SLIDER */ | |
.mtb_slider[type="range"] { | |
-webkit-appearance: none; | |
appearance: none; | |
width: 100%; | |
height: 10px; | |
background: ${S.bgm}; | |
border-radius: 5px; | |
outline: none; | |
opacity: 0.7; | |
transition: opacity .2s; | |
padding: 1em; | |
} | |
/* slider track */ | |
.mtb_slider[type="range"]::-webkit-slider-runnable-track, | |
.mtb_slider[type="range"]::-moz-range-track { | |
width: 100%; | |
height: 10px; | |
background: ${S.bgi}; | |
border-radius: 5px; | |
} | |
/* progress */ | |
.mtb_slider[type="range"]::-moz-range-progress { | |
background-color: ${S.accent}; | |
height:10px; | |
border-radius: 5px; | |
} | |
/* slider thumb (the handle) */ | |
.mtb_slider[type="range"]::-webkit-slider-thumb, | |
.mtb_slider[type="range"]::-moz-range-thumb | |
{ | |
-webkit-appearance: none; | |
appearance: none; | |
width: 15px; | |
height: 15px; | |
border-radius: 50%; | |
background: ${S.fg}; | |
border: none; | |
cursor: pointer; | |
filter: drop-shadow(1px 1px 4px black); | |
} | |
.mtb_slider[type="range"]:focus { | |
opacity: 1; | |
} | |
.mtb_slider[type=range]:-moz-focusring{ | |
outline: 1px solid red; | |
outline-offset: -1px; | |
} | |
.mtb_slider[type="range"]:hover::-webkit-slider-thumb, | |
.mtb_slider[type="range"]:active::-webkit-slider-thumb { | |
background-color: ${S.accent}; | |
} | |
` | |
addNamedStyleSheet( | |
'mtb_ui', | |
` | |
${common} | |
${inputs} | |
`, | |
) | |
} | |
/** | |
* Creates a DOM element with optional styles, class, and id. | |
* | |
* @param {string} kind - The tag name of the element. Supports class and id syntax (e.g. 'div.class#id'). | |
* @param {Object} [style] - CSS styles to apply to the element. | |
* @returns {HTMLElement} - The created DOM element. | |
*/ | |
export const makeElement = (kind, style) => { | |
let [real_kind, className] = kind.split('.') | |
let id | |
if (className?.includes('#')) { | |
;[className, id] = className.split('#') | |
} | |
const el = document.createElement(real_kind) | |
if (style) { | |
Object.assign(el.style, style) | |
} | |
if (className) { | |
el.classList.add(...className.split(' ')) // Support multiple classes | |
} | |
if (id) { | |
el.id = id | |
} | |
return el | |
} | |
/** | |
* Clears all child elements of the given parent element. | |
* | |
* @param {HTMLElement} el - The parent element whose children should be removed. | |
*/ | |
export const clearElement = (el) => { | |
while (el.firstChild) { | |
el.removeChild(el.firstChild) | |
} | |
} | |
/** | |
* Creates a labeled element (input, select, etc.). | |
* | |
* @param {HTMLElement} el - The element to label. | |
* @param {string} labelText - The label text. | |
* @returns {HTMLDivElement} - A div containing the label and the element. | |
*/ | |
export const makeLabeledElement = (el, labelText) => { | |
const wrapper = makeElement('div.mtb_labeled_element', { | |
marginBottom: '1em', | |
}) | |
const label = makeElement('label', { | |
display: 'block', | |
marginBottom: '0.5em', | |
}) | |
label.textContent = labelText | |
wrapper.appendChild(label) | |
wrapper.appendChild(el) | |
return wrapper | |
} | |
/** | |
* Converts a camelCase CSS property to kebab-case. | |
* | |
* @param {string} prop - The camelCase CSS property. | |
* @returns {string} - The kebab-case CSS property. | |
*/ | |
const camelToKebab = (prop) => | |
prop.replace(/[A-Z]/g, (match) => `-${match.toLowerCase()}`) | |
/** | |
* Parses the style string into an object of CSS property-value pairs. | |
* | |
* @param {string} styleString - The CSS rule text (e.g., "color: red; background-color: blue;"). | |
* @returns {Object} - An object with camelCase CSS properties. | |
*/ | |
const parseStyleString = (styleString) => { | |
const styleObj = {} | |
for (const rule of styleString.split(';')) { | |
const [property, value] = rule.split(':').map((item) => item.trim()) | |
if (property && value) { | |
const camelProp = property.replace(/-([a-z])/g, (g) => g[1].toUpperCase()) | |
styleObj[camelProp] = value | |
} | |
} | |
return styleObj | |
} | |
/** | |
* Defines a new CSS class with the provided styles, or skips if the class already exists. | |
* | |
* @param {string} className - The name of the CSS class to define. | |
* @param {Object} classStyles - An object containing camelCase CSS property-value pairs. | |
*/ | |
export function defineCSSClass(className, classStyles) { | |
const styleSheets = document.styleSheets | |
let classExists = false | |
let existingStyleString = '' | |
const classExistsInStyleSheet = (styleSheet) => { | |
const rules = styleSheet.rules || styleSheet.cssRules | |
for (const rule of rules) { | |
if (rule.selectorText === `.${className}`) { | |
classExists = true | |
existingStyleString = rule.style.cssText // Capture existing styles | |
return true | |
} | |
} | |
return false | |
} | |
for (const styleSheet of styleSheets) { | |
if (classExistsInStyleSheet(styleSheet)) { | |
console.debug(`Class ${className} already exists, merging styles...`) | |
break | |
} | |
} | |
const existingStyles = classExists | |
? parseStyleString(existingStyleString) | |
: {} | |
const mergedStyles = { ...existingStyles, ...classStyles } | |
const stylesString = Object.entries(mergedStyles) | |
.map(([key, value]) => `${camelToKebab(key)}: ${value};`) | |
.join(' ') | |
if (!classExists) { | |
console.debug(`Defining new class ${className}...`) | |
if (styleSheets[0].insertRule) { | |
styleSheets[0].insertRule(`.${className} { ${stylesString} }`, 0) | |
} else if (styleSheets[0].addRule) { | |
styleSheets[0].addRule(`.${className}`, stylesString, 0) | |
} | |
} else { | |
console.debug(`Updating existing class ${className} with merged styles...`) | |
for (const styleSheet of styleSheets) { | |
const rules = styleSheet.rules || styleSheet.cssRules | |
for (const rule of rules) { | |
if (rule.selectorText === `.${className}`) { | |
rule.style.cssText = stylesString // Update the existing rule | |
} | |
} | |
} | |
} | |
console.debug( | |
`Class ${className} has been defined/updated with styles:`, | |
mergedStyles, | |
) | |
} | |
/** | |
* Renders a sidebar and ensures it resizes correctly when the window is resized. | |
* | |
* @param {HTMLElement} el - The element where the sidebar is rendered. | |
* @param {HTMLElement} cont - The content container of the sidebar. | |
* @param {HTMLElement[]} elems - Array of elements to append to the sidebar. | |
* @returns {Object} - A handle with a method to unregister the resize event. | |
*/ | |
export const renderSidebar = (el, cont, elems) => { | |
el.appendChild(cont) | |
if (!el.parentNode) { | |
return | |
} | |
el.parentNode.style.overflowY = 'clip' | |
cont.style.height = `${el.parentNode.offsetHeight}px` | |
const resizeHandler = () => { | |
cont.style.height = `${el.parentNode.offsetHeight}px` | |
} | |
window.addEventListener('resize', resizeHandler) | |
for (const elem of elems) { | |
cont.appendChild(elem) | |
} | |
return { | |
unregister: () => { | |
window.removeEventListener('resize', resizeHandler) | |
}, | |
} | |
} | |
/** | |
* Creates a <select> dropdown with given options. | |
* | |
* @param {string[]} options - The options for the select element. | |
* @param {string} [current] - The currently selected option (optional). | |
* @returns {HTMLSelectElement} - The created <select> element. | |
*/ | |
export const makeSelect = (options, current = undefined) => { | |
const selector = makeElement('select.mtb_select', { | |
width: 'auto', | |
margin: '1em', | |
}) | |
for (const option of options) { | |
const opt = makeElement('option') | |
opt.value = option | |
opt.innerHTML = option | |
selector.appendChild(opt) | |
} | |
if (current !== undefined) { | |
if (options.includes(current)) { | |
selector.value = current | |
} else { | |
console.error( | |
`You tried to select an option that doesn't exist (${current}). Options: ${options}`, | |
) | |
} | |
} | |
return selector | |
} | |
/** | |
* Creates an <input type="range"> slider element with given parameters. | |
* | |
* @param {number} min - Minimum value of the slider. | |
* @param {number} max - Maximum value of the slider. | |
* @param {number} [value] - Initial value of the slider. | |
* @param {number} [step] - Step value for the slider. | |
* @returns {HTMLInputElement} - The created slider element. | |
*/ | |
export const makeSlider = (min, max, value = undefined, step = undefined) => { | |
const slider = makeElement('input.mtb_slider', { | |
width: '100%', | |
}) | |
slider.type = 'range' | |
slider.min = min || 0 | |
slider.max = max || 100 | |
slider.value = value || slider.min | |
slider.step = step || 1 | |
return slider | |
} | |
/** | |
* Creates a button element. | |
* | |
* @param {string} label - The label for the button. | |
* @param {Object} [style] - Optional styles to apply to the button. | |
* @param {Function} [onClick] - Optional click handler. | |
* @returns {HTMLButtonElement} - The created button element. | |
*/ | |
export const makeButton = (label, style = {}, onClick = undefined) => { | |
const button = makeElement('button.mtb_button', style) | |
button.textContent = label | |
if (onClick) { | |
button.addEventListener('click', onClick) | |
} | |
return button | |
} | |
/** | |
* Creates a resizable splitter between two elements. | |
* | |
* @param {HTMLElement} el1 - The first element. | |
* @param {HTMLElement} el2 - The second element. | |
* @param {'vertical' | 'horizontal'} direction - Splitter direction (vertical or horizontal). | |
* @param {'absolute' | 'normal'} mode - Splitter mode: 'absolute' for free resizing, 'normal' for layout-based resizing. | |
* @returns {HTMLDivElement} - The container with resizable splitter. | |
*/ | |
export const makeSplitter = ( | |
el1, | |
el2, | |
direction = 'vertical', | |
mode = 'normal', | |
) => { | |
const container = makeElement('div.mtb_splitter_container', { | |
display: mode === 'absolute' ? 'block' : 'flex', | |
flexDirection: direction === 'vertical' ? 'row' : 'column', | |
position: mode === 'absolute' ? 'relative' : 'static', | |
height: '100%', | |
width: '100%', | |
}) | |
const handle = makeElement('div.mtb_splitter_handle', { | |
backgroundColor: '#ccc', | |
cursor: direction === 'vertical' ? 'col-resize' : 'row-resize', | |
width: direction === 'vertical' ? '5px' : '100%', | |
height: direction === 'horizontal' ? '5px' : '100%', | |
}) | |
let isResizing = false | |
handle.addEventListener('mousedown', () => { | |
isResizing = true | |
}) | |
window.addEventListener('mouseup', () => { | |
isResizing = false | |
}) | |
window.addEventListener('mousemove', (e) => { | |
if (!isResizing) return | |
if (direction === 'vertical') { | |
const newWidth = e.clientX - container.offsetLeft | |
el1.style.width = `${newWidth}px` | |
el2.style.width = `${container.offsetWidth - newWidth}px` | |
} else { | |
const newHeight = e.clientY - container.offsetTop | |
el1.style.height = `${newHeight}px` | |
el2.style.height = `${container.offsetHeight - newHeight}px` | |
} | |
}) | |
container.appendChild(el1) | |
container.appendChild(handle) | |
container.appendChild(el2) | |
return container | |
} | |