fix(ui): keep autocomplete text widget at max-height on node resize in Vue mode

This commit is contained in:
Will Miao
2026-06-08 10:49:04 +08:00
parent 568daa351e
commit 00177a06d0
4 changed files with 149 additions and 16 deletions

View File

@@ -546,6 +546,27 @@ function normalizeAutocompleteWidgetValues(node: any, info: any) {
}
}
function applyAutocompleteTextLayoutFix(
widget: any,
container: HTMLElement | undefined,
isVueMode: boolean
): void {
if (isVueMode) {
;(widget as any).computeLayoutSize = undefined
widget.computeSize = (width?: number) =>
[width ?? 200, AUTOCOMPLETE_TEXT_WIDGET_MAX_HEIGHT - 4]
if (container) {
container.style.minHeight = `${AUTOCOMPLETE_TEXT_WIDGET_MAX_HEIGHT}px`
}
} else {
delete (widget as any).computeLayoutSize
delete (widget as any).computeSize
if (container) {
container.style.minHeight = ''
}
}
}
// Listen for Vue DOM mode setting changes and dispatch custom event
const initVueDomModeListener = () => {
if (app.ui?.settings?.addEventListener) {
@@ -554,7 +575,47 @@ const initVueDomModeListener = () => {
// before we read it (the event may fire before internal state updates)
requestAnimationFrame(() => {
const isVueDomMode = app.ui?.settings?.getSettingValue?.('Comfy.VueNodes.Enabled') ?? false
// Dispatch custom event for Vue components to listen to
if (app.graph?.nodes) {
for (const node of app.graph.nodes) {
const textWidget = node.widgets?.find(
(w: any) => w.type === 'AUTOCOMPLETE_TEXT_LORAS'
)
if (!textWidget) continue
const container = (textWidget as any).element as HTMLElement | undefined
applyAutocompleteTextLayoutFix(textWidget, container, isVueDomMode)
}
}
requestAnimationFrame(() => {
for (const nodeEl of document.querySelectorAll('[data-node-id]')) {
const grid = nodeEl.querySelector('[data-testid="node-widgets"]') as HTMLElement | null
if (!grid) continue
const nodeId = nodeEl.getAttribute('data-node-id')
const node = app.graph?.getNodeById(nodeId as any)
if (!node) continue
const rows: string[] = []
let needsFix = false
for (const w of node.widgets ?? []) {
if (w.type === 'LORA_MANAGER_AUTOCOMPLETE_METADATA') {
rows.push('min-content')
} else if (w.name === 'loras') {
rows.push('auto')
} else if (w.name === 'text' && w.type === 'AUTOCOMPLETE_TEXT_LORAS') {
rows.push(isVueDomMode ? 'min-content' : 'auto')
needsFix = true
} else {
rows.push('auto')
}
}
if (needsFix) {
grid.style.gridTemplateRows = rows.join(' ')
}
}
})
app.canvas?.setDirty(true, true)
document.dispatchEvent(new CustomEvent('lora-manager:vue-mode-change', {
detail: { isVueDomMode }
}))
@@ -678,6 +739,15 @@ function createAutocompleteTextWidgetFactory(
if (maxHeight) {
container.style.maxHeight = `${maxHeight}px`
container.style.minHeight = `${maxHeight}px`
}
if (modelType === 'loras') {
applyAutocompleteTextLayoutFix(
widget,
container,
typeof LiteGraph !== 'undefined' && LiteGraph.vueNodesMode
)
}
widget.onRemove = createVueWidgetCleanup(vueApp, () => {