14 Commits

Author SHA1 Message Date
8911323832 Branch-grouped navigation for timeline node manager
Replace flat dropdown with branch selector showing node counts,
scrollable node list with HEAD/tip badges, and inline actions panel.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-27 22:15:56 +01:00
af5eafaf4d Right-align path inputs to show filename instead of directory prefix
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-27 00:02:14 +01:00
29750acf58 Match Shift button height to input field
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:28:35 +01:00
da789e68ad Two-column VACE layout, inline mode reference button
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:26:45 +01:00
79755c286b Move VACE Settings to full-width section below splitter columns
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:23:37 +01:00
39a1b98924 Fix history snapshot corruption, missing dir crash, stale batch delete
- Deep-copy node data on restore to prevent edits from mutating
  stored history snapshots
- Guard glob calls against non-existent current_dir
- Read current selection at delete time instead of using stale
  render-time capture

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:11:11 +01:00
d3dbd4645a Remove Promote button (legacy single-file editor feature)
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:05:19 +01:00
d795671763 Display LoRA strength with one decimal place (1.0 not 1)
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:03:56 +01:00
9f141ba42f Fix input sync bugs, improve LoRA UX, and harden edge cases
- Sync dict_input/dict_textarea/LoRA inputs on update:model-value
  (not just blur) to prevent silent data loss on quick saves
- Split LoRA into name + strength fields, default strength to 1.0
- Stack LoRAs one per line instead of 3-card row
- Collapse "Add New Sequence from Source File" into expansion
- Add file selector to Pane A in dual-pane mode
- Clear secondary pane state on directory change
- Fix file radio resetting to first file on refresh
- Handle bare-list JSON files and inf/nan edge cases

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 18:02:24 +01:00
7931060d43 Fix number inputs not syncing to dict until blur
dict_number() only wrote to seq[key] on blur, so changing a value
(e.g. via spinner arrows) and immediately clicking Save could race
the save ahead of the blur on the server. Now also syncs on
update:model-value so the dict is always current.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 17:49:09 +01:00
3264845e68 Add dual-pane batch processor with independent file state
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 17:41:25 +01:00
fe2c6445ef Constrain main content area to 1200px max-width
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 17:29:08 +01:00
710a8407d2 Overhaul UI: new color palette, spacing, and visual hierarchy
Replace red accent with amber, add Inter font, introduce 4-level depth
palette via CSS variables, expand padding/gaps, wrap sidebar and content
sections in cards, add section/subsection header typography classes, and
style scrollbars for dark theme. Pure visual changes — no functional or
data-flow modifications.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 17:27:02 +01:00
97748ab8ff Fix VACE schedule default mismatch introduced in refactor
dict_number() defaulted to 0 while mode_label used default of 1,
causing visual inconsistency when 'vace schedule' key is missing.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-26 17:01:09 +01:00
6 changed files with 895 additions and 605 deletions

512
main.py
View File

@@ -20,72 +20,136 @@ from tab_comfy_ng import render_comfy_monitor
def index(): def index():
# -- Streamlit dark theme -- # -- Streamlit dark theme --
ui.dark_mode(True) ui.dark_mode(True)
ui.colors(primary='#FF4B4B') ui.colors(primary='#F59E0B')
ui.add_head_html(
'<link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap">'
)
ui.add_css(''' ui.add_css('''
/* === Streamlit Dark Theme === */ /* === Dark Theme with Depth Palette === */
:root {
--bg-page: #0B0E14;
--bg-surface-1: #13161E;
--bg-surface-2: #1A1E2A;
--bg-surface-3: #242836;
--border: rgba(255,255,255,0.08);
--text-primary: #EAECF0;
--text-secondary: rgba(234,236,240,0.55);
--accent: #F59E0B;
--accent-subtle: rgba(245,158,11,0.12);
--negative: #EF4444;
}
/* Backgrounds */ /* Backgrounds */
body.body--dark, body.body--dark,
.q-page.body--dark, .q-page.body--dark,
.body--dark .q-page { background: #0E1117 !important; } .body--dark .q-page { background: var(--bg-page) !important; }
.body--dark .q-drawer { background: #262730 !important; } .body--dark .q-drawer { background: var(--bg-surface-1) !important; }
.body--dark .q-card { background: #262730 !important; border-radius: 0.5rem; } .body--dark .q-card {
background: var(--bg-surface-2) !important;
border: 1px solid var(--border);
border-radius: 0.75rem;
}
.body--dark .q-tab-panels { background: transparent !important; } .body--dark .q-tab-panels { background: transparent !important; }
.body--dark .q-tab-panel { background: transparent !important; } .body--dark .q-tab-panel { background: transparent !important; }
.body--dark .q-expansion-item { background: transparent !important; } .body--dark .q-expansion-item { background: transparent !important; }
/* Text */ /* Text */
.body--dark { color: #FAFAFA !important; } .body--dark { color: var(--text-primary) !important; }
.body--dark .q-field__label { color: rgba(250,250,250,0.6) !important; } .body--dark .q-field__label { color: var(--text-secondary) !important; }
.body--dark .text-caption { color: rgba(250,250,250,0.6) !important; } .body--dark .text-caption { color: var(--text-secondary) !important; }
.body--dark .text-subtitle1, .body--dark .text-subtitle1,
.body--dark .text-subtitle2 { color: #FAFAFA !important; } .body--dark .text-subtitle2 { color: var(--text-primary) !important; }
/* Inputs & textareas */ /* Inputs & textareas */
.body--dark .q-field--outlined .q-field__control { .body--dark .q-field--outlined .q-field__control {
background: #262730 !important; background: var(--bg-surface-3) !important;
border-radius: 0.5rem !important; border-radius: 0.5rem !important;
} }
.body--dark .q-field--outlined .q-field__control:before { .body--dark .q-field--outlined .q-field__control:before {
border-color: rgba(250,250,250,0.2) !important; border-color: var(--border) !important;
border-radius: 0.5rem !important; border-radius: 0.5rem !important;
} }
.body--dark .q-field--outlined.q-field--focused .q-field__control:after { .body--dark .q-field--outlined.q-field--focused .q-field__control:after {
border-color: #FF4B4B !important; border-color: var(--accent) !important;
} }
.body--dark .q-field__native, .body--dark .q-field__native,
.body--dark .q-field__input { color: #FAFAFA !important; } .body--dark .q-field__input { color: var(--text-primary) !important; }
/* Sidebar inputs get main bg */ /* Sidebar inputs get page bg */
.body--dark .q-drawer .q-field--outlined .q-field__control { .body--dark .q-drawer .q-field--outlined .q-field__control {
background: #0E1117 !important; background: var(--bg-page) !important;
} }
/* Buttons */ /* Buttons */
.body--dark .q-btn--standard { border-radius: 0.5rem !important; } .body--dark .q-btn--standard { border-radius: 0.5rem !important; }
.body--dark .q-btn--outline {
transition: background 0.15s ease;
}
.body--dark .q-btn--outline:hover {
background: var(--accent-subtle) !important;
}
/* Tabs */ /* Tabs */
.body--dark .q-tab--active { color: #FF4B4B !important; } .body--dark .q-tab--active { color: var(--accent) !important; }
.body--dark .q-tab__indicator { background: #FF4B4B !important; } .body--dark .q-tab__indicator { background: var(--accent) !important; }
/* Separators */ /* Separators */
.body--dark .q-separator { background: rgba(250,250,250,0.2) !important; } .body--dark .q-separator { background: var(--border) !important; }
/* Expansion items */ /* Expansion items */
.body--dark .q-expansion-item__content { padding: 4px 0; } .body--dark .q-expansion-item__content { padding: 12px 16px; }
.body--dark .q-item { border-radius: 0.5rem; } .body--dark .q-item { border-radius: 0.5rem; }
/* Splitter */ /* Splitter */
.body--dark .q-splitter__separator { background: rgba(250,250,250,0.2) !important; } .body--dark .q-splitter__separator { background: var(--border) !important; }
.body--dark .q-splitter__before,
.body--dark .q-splitter__after { padding: 0 8px; }
/* Action row wrap */ /* Action row wrap */
.action-row { flex-wrap: wrap !important; gap: 4px !important; } .action-row { flex-wrap: wrap !important; gap: 8px !important; }
/* Notifications */ /* Notifications */
.body--dark .q-notification { border-radius: 0.5rem; } .body--dark .q-notification { border-radius: 0.5rem; }
/* Font */ /* Font */
body { font-family: "Source Sans Pro", "Source Sans 3", sans-serif !important; } body { font-family: "Inter", "Source Sans Pro", "Source Sans 3", sans-serif !important; }
/* Surface utility classes (need .body--dark to beat .body--dark .q-card specificity) */
.body--dark .surface-1 { background: var(--bg-surface-1) !important; }
.body--dark .surface-2 { background: var(--bg-surface-2) !important; }
.body--dark .surface-3 { background: var(--bg-surface-3) !important; }
/* Typography utility classes */
.section-header {
font-size: 0.8rem;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.05em;
color: var(--text-secondary) !important;
}
.subsection-header {
font-size: 0.85rem;
font-weight: 500;
color: var(--text-primary) !important;
}
/* Scrollbar */
::-webkit-scrollbar { width: 6px; height: 6px; }
::-webkit-scrollbar-track { background: transparent; }
::-webkit-scrollbar-thumb {
background: rgba(255,255,255,0.12);
border-radius: 3px;
}
::-webkit-scrollbar-thumb:hover {
background: rgba(255,255,255,0.2);
}
/* Secondary pane teal accent */
.pane-secondary .q-field--outlined.q-field--focused .q-field__control:after {
border-color: #06B6D4 !important;
}
.pane-secondary .q-btn.bg-primary { background-color: #06B6D4 !important; }
.pane-secondary .section-header { color: rgba(6,182,212,0.7) !important; }
''') ''')
config = load_config() config = load_config()
@@ -94,6 +158,7 @@ def index():
current_dir=Path(config.get('last_dir', str(Path.cwd()))), current_dir=Path(config.get('last_dir', str(Path.cwd()))),
snippets=load_snippets(), snippets=load_snippets(),
) )
dual_pane = {'active': False, 'state': None}
# ------------------------------------------------------------------ # ------------------------------------------------------------------
# Define helpers FIRST (before sidebar, which needs them) # Define helpers FIRST (before sidebar, which needs them)
@@ -101,30 +166,92 @@ def index():
@ui.refreshable @ui.refreshable
def render_main_content(): def render_main_content():
if not state.file_path or not state.file_path.exists(): max_w = '2400px' if dual_pane['active'] else '1200px'
ui.label('Select a file from the sidebar to begin.').classes( with ui.column().classes('w-full q-pa-md').style(f'max-width: {max_w}; margin: 0 auto'):
'text-subtitle1 q-pa-lg') if not state.file_path or not state.file_path.exists():
ui.label('Select a file from the sidebar to begin.').classes(
'text-subtitle1 q-pa-lg')
return
ui.label(f'Editing: {state.file_path.name}').classes('text-h5 q-mb-lg').style('font-weight: 600')
with ui.tabs().classes('w-full').style('border-bottom: 1px solid var(--border)') as tabs:
ui.tab('batch', label='Batch Processor')
ui.tab('timeline', label='Timeline')
ui.tab('raw', label='Raw Editor')
with ui.tab_panels(tabs, value='batch').classes('w-full'):
with ui.tab_panel('batch'):
_render_batch_tab_content()
with ui.tab_panel('timeline'):
render_timeline_tab(state)
with ui.tab_panel('raw'):
render_raw_editor(state)
if state.show_comfy_monitor:
ui.separator()
with ui.expansion('ComfyUI Monitor', icon='dns').classes('w-full'):
render_comfy_monitor(state)
@ui.refreshable
def _render_batch_tab_content():
def on_toggle(e):
dual_pane['active'] = e.value
if e.value and dual_pane['state'] is None:
s2 = state.create_secondary()
s2._render_main = _render_batch_tab_content
dual_pane['state'] = s2
render_main_content.refresh()
ui.switch('Dual Pane', value=dual_pane['active'], on_change=on_toggle)
if not dual_pane['active']:
render_batch_processor(state)
else:
s2 = dual_pane['state']
with ui.row().classes('w-full gap-4'):
with ui.column().classes('col'):
ui.label('Pane A').classes('section-header q-mb-sm')
_render_pane_file_selector(state)
render_batch_processor(state)
with ui.column().classes('col pane-secondary'):
ui.label('Pane B').classes('section-header q-mb-sm')
_render_pane_file_selector(s2)
if s2.file_path and s2.file_path.exists():
render_batch_processor(s2)
else:
ui.label('Select a file above to begin.').classes(
'text-caption q-pa-md')
def _render_pane_file_selector(pane_state: AppState):
if not pane_state.current_dir.exists():
ui.label('Directory not found.').classes('text-warning')
return return
json_files = sorted(pane_state.current_dir.glob('*.json'))
json_files = [f for f in json_files if f.name not in (
'.editor_config.json', '.editor_snippets.json')]
file_names = [f.name for f in json_files]
ui.label(f'Editing: {state.file_path.name}').classes('text-h5 q-mb-md') current_val = pane_state.file_path.name if pane_state.file_path else None
with ui.tabs().classes('w-full') as tabs: def on_select(e):
ui.tab('batch', label='Batch Processor') if not e.value:
ui.tab('timeline', label='Timeline') return
ui.tab('raw', label='Raw Editor') fp = pane_state.current_dir / e.value
data, mtime = load_json(fp)
pane_state.data_cache = data
pane_state.last_mtime = mtime
pane_state.loaded_file = str(fp)
pane_state.file_path = fp
pane_state.restored_indicator = None
_render_batch_tab_content.refresh()
with ui.tab_panels(tabs, value='batch').classes('w-full'): ui.select(
with ui.tab_panel('batch'): file_names,
render_batch_processor(state) value=current_val,
with ui.tab_panel('timeline'): label='File',
render_timeline_tab(state) on_change=on_select,
with ui.tab_panel('raw'): ).classes('w-full')
render_raw_editor(state)
if state.show_comfy_monitor:
ui.separator()
with ui.expansion('ComfyUI Monitor', icon='dns').classes('w-full'):
render_comfy_monitor(state)
def load_file(file_name: str): def load_file(file_name: str):
"""Load a JSON file and refresh the main content.""" """Load a JSON file and refresh the main content."""
@@ -148,8 +275,8 @@ def index():
# ------------------------------------------------------------------ # ------------------------------------------------------------------
# Sidebar (rendered AFTER helpers are attached) # Sidebar (rendered AFTER helpers are attached)
# ------------------------------------------------------------------ # ------------------------------------------------------------------
with ui.left_drawer(value=True).classes('q-pa-md').style('width: 300px'): with ui.left_drawer(value=True).classes('q-pa-md').style('width: 320px'):
render_sidebar(state) render_sidebar(state, dual_pane)
# ------------------------------------------------------------------ # ------------------------------------------------------------------
# Main content area # Main content area
@@ -162,175 +289,190 @@ def index():
# Sidebar # Sidebar
# ====================================================================== # ======================================================================
def render_sidebar(state: AppState): def render_sidebar(state: AppState, dual_pane: dict):
ui.label('Navigator').classes('text-h6') ui.label('Navigator').classes('text-h6')
# --- Path input --- # --- Path input + Pin ---
path_input = ui.input( with ui.card().classes('w-full q-pa-md q-mb-md'):
'Current Path', path_input = ui.input(
value=str(state.current_dir), 'Current Path',
).classes('w-full') value=str(state.current_dir),
).classes('w-full')
def on_path_enter(): def on_path_enter():
p = resolve_path_case_insensitive(path_input.value) p = resolve_path_case_insensitive(path_input.value)
if p is not None and p.is_dir(): if p is not None and p.is_dir():
state.current_dir = p state.current_dir = p
state.config['last_dir'] = str(p) if dual_pane['state']:
dual_pane['state'].current_dir = state.current_dir
dual_pane['state'].file_path = None
dual_pane['state'].loaded_file = None
dual_pane['state'].data_cache = {}
state.config['last_dir'] = str(p)
save_config(state.current_dir, state.config['favorites'], state.config)
state.loaded_file = None
state.file_path = None
path_input.set_value(str(p))
render_file_list.refresh()
# Auto-load inside render_file_list already refreshed main content
# if files exist; only refresh here for the empty-directory case.
if not state.loaded_file:
state._render_main.refresh()
path_input.on('keydown.enter', lambda _: on_path_enter())
def pin_folder():
d = str(state.current_dir)
if d not in state.config['favorites']:
state.config['favorites'].append(d)
save_config(state.current_dir, state.config['favorites'], state.config)
render_favorites.refresh()
ui.button('Pin Folder', icon='push_pin', on_click=pin_folder).classes('w-full')
# --- Favorites ---
with ui.card().classes('w-full q-pa-md q-mb-md'):
ui.label('Favorites').classes('section-header')
@ui.refreshable
def render_favorites():
for fav in list(state.config['favorites']):
with ui.row().classes('w-full items-center'):
ui.button(
fav,
on_click=lambda f=fav: _jump_to(f),
).props('flat dense').classes('col')
ui.button(
icon='close',
on_click=lambda f=fav: _unpin(f),
).props('flat dense color=negative')
def _jump_to(fav: str):
state.current_dir = Path(fav)
if dual_pane['state']:
dual_pane['state'].current_dir = state.current_dir
dual_pane['state'].file_path = None
dual_pane['state'].loaded_file = None
dual_pane['state'].data_cache = {}
state.config['last_dir'] = fav
save_config(state.current_dir, state.config['favorites'], state.config) save_config(state.current_dir, state.config['favorites'], state.config)
state.loaded_file = None state.loaded_file = None
state.file_path = None state.file_path = None
path_input.set_value(str(p)) path_input.set_value(fav)
render_file_list.refresh() render_file_list.refresh()
# Auto-load inside render_file_list already refreshed main content
# if files exist; only refresh here for the empty-directory case.
if not state.loaded_file: if not state.loaded_file:
state._render_main.refresh() state._render_main.refresh()
path_input.on('keydown.enter', lambda _: on_path_enter()) def _unpin(fav: str):
if fav in state.config['favorites']:
state.config['favorites'].remove(fav)
save_config(state.current_dir, state.config['favorites'], state.config)
render_favorites.refresh()
# --- Pin / Unpin --- render_favorites()
def pin_folder():
d = str(state.current_dir)
if d not in state.config['favorites']:
state.config['favorites'].append(d)
save_config(state.current_dir, state.config['favorites'], state.config)
render_favorites.refresh()
ui.button('Pin Folder', icon='push_pin', on_click=pin_folder).classes('w-full')
@ui.refreshable
def render_favorites():
for fav in list(state.config['favorites']):
with ui.row().classes('w-full items-center'):
ui.button(
fav,
on_click=lambda f=fav: _jump_to(f),
).props('flat dense').classes('col')
ui.button(
icon='close',
on_click=lambda f=fav: _unpin(f),
).props('flat dense color=negative')
def _jump_to(fav: str):
state.current_dir = Path(fav)
state.config['last_dir'] = fav
save_config(state.current_dir, state.config['favorites'], state.config)
state.loaded_file = None
state.file_path = None
path_input.set_value(fav)
render_file_list.refresh()
if not state.loaded_file:
state._render_main.refresh()
def _unpin(fav: str):
if fav in state.config['favorites']:
state.config['favorites'].remove(fav)
save_config(state.current_dir, state.config['favorites'], state.config)
render_favorites.refresh()
render_favorites()
ui.separator()
# --- Snippet Library --- # --- Snippet Library ---
ui.label('Snippet Library').classes('text-subtitle1 q-mt-md') with ui.card().classes('w-full q-pa-md q-mb-md'):
ui.label('Snippet Library').classes('section-header')
with ui.expansion('Add New Snippet'): with ui.expansion('Add New Snippet'):
snip_name_input = ui.input('Name', placeholder='e.g. Cinematic').classes('w-full') snip_name_input = ui.input('Name', placeholder='e.g. Cinematic').classes('w-full')
snip_content_input = ui.textarea('Content', placeholder='4k, high quality...').classes('w-full') snip_content_input = ui.textarea('Content', placeholder='4k, high quality...').classes('w-full')
def save_snippet(): def save_snippet():
name = snip_name_input.value name = snip_name_input.value
content = snip_content_input.value content = snip_content_input.value
if name and content: if name and content:
state.snippets[name] = content state.snippets[name] = content
save_snippets(state.snippets)
snip_name_input.set_value('')
snip_content_input.set_value('')
ui.notify(f"Saved '{name}'")
render_snippet_list.refresh()
ui.button('Save Snippet', on_click=save_snippet).classes('w-full')
@ui.refreshable
def render_snippet_list():
if not state.snippets:
return
ui.label('Click to copy snippet text:').classes('text-caption')
for name, content in list(state.snippets.items()):
with ui.row().classes('w-full items-center'):
async def copy_snippet(c=content):
await ui.run_javascript(
f'navigator.clipboard.writeText({json.dumps(c)})', timeout=3.0)
ui.notify('Copied to clipboard')
ui.button(
f'{name}',
on_click=copy_snippet,
).props('flat dense').classes('col')
ui.button(
icon='delete',
on_click=lambda n=name: _del_snippet(n),
).props('flat dense color=negative')
def _del_snippet(name: str):
if name in state.snippets:
del state.snippets[name]
save_snippets(state.snippets) save_snippets(state.snippets)
snip_name_input.set_value('')
snip_content_input.set_value('')
ui.notify(f"Saved '{name}'")
render_snippet_list.refresh() render_snippet_list.refresh()
ui.button('Save Snippet', on_click=save_snippet).classes('w-full') render_snippet_list()
@ui.refreshable
def render_snippet_list():
if not state.snippets:
return
ui.label('Click to copy snippet text:').classes('text-caption')
for name, content in list(state.snippets.items()):
with ui.row().classes('w-full items-center'):
async def copy_snippet(c=content):
await ui.run_javascript(
f'navigator.clipboard.writeText({json.dumps(c)})', timeout=3.0)
ui.notify('Copied to clipboard')
ui.button(
f'{name}',
on_click=copy_snippet,
).props('flat dense').classes('col')
ui.button(
icon='delete',
on_click=lambda n=name: _del_snippet(n),
).props('flat dense color=negative')
def _del_snippet(name: str):
if name in state.snippets:
del state.snippets[name]
save_snippets(state.snippets)
render_snippet_list.refresh()
render_snippet_list()
ui.separator()
# --- File List --- # --- File List ---
@ui.refreshable with ui.card().classes('w-full q-pa-md q-mb-md'):
def render_file_list(): @ui.refreshable
json_files = sorted(state.current_dir.glob('*.json')) def render_file_list():
json_files = [f for f in json_files if f.name not in ('.editor_config.json', '.editor_snippets.json')] if not state.current_dir.exists():
ui.label('Directory not found.').classes('text-warning')
return
json_files = sorted(state.current_dir.glob('*.json'))
json_files = [f for f in json_files if f.name not in ('.editor_config.json', '.editor_snippets.json')]
if not json_files: if not json_files:
ui.label('No JSON files in this folder.').classes('text-caption') ui.label('No JSON files in this folder.').classes('text-caption')
ui.button('Generate Templates', on_click=lambda: _gen_templates()).classes('w-full') ui.button('Generate Templates', on_click=lambda: _gen_templates()).classes('w-full')
return return
with ui.expansion('Create New JSON'): with ui.expansion('Create New JSON'):
new_fn_input = ui.input('Filename', placeholder='my_prompt_vace').classes('w-full') new_fn_input = ui.input('Filename', placeholder='my_prompt_vace').classes('w-full')
def create_new(): def create_new():
fn = new_fn_input.value fn = new_fn_input.value
if not fn: if not fn:
return return
if not fn.endswith('.json'): if not fn.endswith('.json'):
fn += '.json' fn += '.json'
path = state.current_dir / fn path = state.current_dir / fn
first_item = DEFAULTS.copy() first_item = DEFAULTS.copy()
first_item[KEY_SEQUENCE_NUMBER] = 1 first_item[KEY_SEQUENCE_NUMBER] = 1
save_json(path, {KEY_BATCH_DATA: [first_item]}) save_json(path, {KEY_BATCH_DATA: [first_item]})
new_fn_input.set_value('') new_fn_input.set_value('')
render_file_list.refresh() render_file_list.refresh()
ui.button('Create', on_click=create_new).classes('w-full') ui.button('Create', on_click=create_new).classes('w-full')
ui.label('Select File').classes('text-subtitle2 q-mt-sm') ui.label('Select File').classes('subsection-header q-mt-sm')
file_names = [f.name for f in json_files] file_names = [f.name for f in json_files]
ui.radio( current = Path(state.loaded_file).name if state.loaded_file else None
file_names, selected = current if current in file_names else (file_names[0] if file_names else None)
value=file_names[0] if file_names else None, ui.radio(
on_change=lambda e: state._load_file(e.value) if e.value else None, file_names,
).classes('w-full') value=selected,
on_change=lambda e: state._load_file(e.value) if e.value else None,
).classes('w-full')
# Auto-load first file if nothing loaded yet # Auto-load first file if nothing loaded yet
if file_names and not state.loaded_file: if file_names and not state.loaded_file:
state._load_file(file_names[0]) state._load_file(file_names[0])
def _gen_templates(): def _gen_templates():
generate_templates(state.current_dir) generate_templates(state.current_dir)
render_file_list.refresh() render_file_list.refresh()
render_file_list() render_file_list()
ui.separator()
# --- Comfy Monitor toggle --- # --- Comfy Monitor toggle ---
def on_monitor_toggle(e): def on_monitor_toggle(e):

View File

@@ -23,3 +23,10 @@ class AppState:
_main_rendered: bool = False _main_rendered: bool = False
_live_checkboxes: dict = field(default_factory=dict) _live_checkboxes: dict = field(default_factory=dict)
_live_refreshables: dict = field(default_factory=dict) _live_refreshables: dict = field(default_factory=dict)
def create_secondary(self) -> 'AppState':
return AppState(
config=self.config,
current_dir=self.current_dir,
snippets=self.snippets,
)

View File

@@ -79,42 +79,56 @@ def find_insert_position(batch_list, parent_index, parent_seq_num):
# --- Helper for repetitive dict-bound inputs --- # --- Helper for repetitive dict-bound inputs ---
def dict_input(element_fn, label, seq, key, **kwargs): def dict_input(element_fn, label, seq, key, **kwargs):
"""Create an input element bound to seq[key] via blur event.""" """Create an input element bound to seq[key] via blur and model-value update."""
val = seq.get(key, '') val = seq.get(key, '')
if isinstance(val, (int, float)): if isinstance(val, (int, float)):
val = str(val) if element_fn != ui.number else val val = str(val) if element_fn != ui.number else val
el = element_fn(label, value=val, **kwargs) el = element_fn(label, value=val, **kwargs)
el.on('blur', lambda e, k=key: seq.__setitem__(k, e.sender.value))
def _sync(k=key):
seq[k] = el.value
el.on('blur', lambda _: _sync())
el.on('update:model-value', lambda _: _sync())
return el return el
def dict_number(label, seq, key, default=0, **kwargs): def dict_number(label, seq, key, default=0, **kwargs):
"""Number input bound to seq[key] via blur.""" """Number input bound to seq[key] via blur and model-value update."""
val = seq.get(key, default) val = seq.get(key, default)
try: try:
# Try float first to handle "1.5" strings, then check if it's a clean int # Try float first to handle "1.5" strings, then check if it's a clean int
fval = float(val) fval = float(val)
val = int(fval) if fval == int(fval) else fval val = int(fval) if fval == int(fval) else fval
except (ValueError, TypeError): except (ValueError, TypeError, OverflowError):
val = default val = default
el = ui.number(label, value=val, **kwargs) el = ui.number(label, value=val, **kwargs)
def _on_blur(e, k=key, d=default): def _sync(k=key, d=default):
v = e.sender.value v = el.value
if v is None: if v is None:
v = d v = d
elif isinstance(v, float) and v == int(v): elif isinstance(v, float):
v = int(v) try:
v = int(v) if v == int(v) else v
except (OverflowError, ValueError):
v = d
seq[k] = v seq[k] = v
el.on('blur', _on_blur) el.on('blur', lambda _: _sync())
el.on('update:model-value', lambda _: _sync())
return el return el
def dict_textarea(label, seq, key, **kwargs): def dict_textarea(label, seq, key, **kwargs):
"""Textarea bound to seq[key] via blur.""" """Textarea bound to seq[key] via blur and model-value update."""
el = ui.textarea(label, value=seq.get(key, ''), **kwargs) el = ui.textarea(label, value=seq.get(key, ''), **kwargs)
el.on('blur', lambda e, k=key: seq.__setitem__(k, e.sender.value))
def _sync(k=key):
seq[k] = el.value
el.on('blur', lambda _: _sync())
el.on('update:model-value', lambda _: _sync())
return el return el
@@ -125,7 +139,10 @@ def dict_textarea(label, seq, key, **kwargs):
def render_batch_processor(state: AppState): def render_batch_processor(state: AppState):
data = state.data_cache data = state.data_cache
file_path = state.file_path file_path = state.file_path
is_batch_file = KEY_BATCH_DATA in data or isinstance(data, list) if isinstance(data, list):
data = {KEY_BATCH_DATA: data}
state.data_cache = data
is_batch_file = KEY_BATCH_DATA in data
if not is_batch_file: if not is_batch_file:
ui.label('This is a Single file. To use Batch mode, create a copy.').classes( ui.label('This is a Single file. To use Batch mode, create a copy.').classes(
@@ -156,69 +173,66 @@ def render_batch_processor(state: AppState):
batch_list = data.get(KEY_BATCH_DATA, []) batch_list = data.get(KEY_BATCH_DATA, [])
# Source file data for importing # Source file data for importing
json_files = sorted(state.current_dir.glob('*.json')) with ui.card().classes('w-full q-pa-md q-mb-lg'):
json_files = [f for f in json_files if f.name not in ( with ui.expansion('Add New Sequence from Source File', icon='playlist_add').classes('w-full'):
'.editor_config.json', '.editor_snippets.json')] json_files = sorted(state.current_dir.glob('*.json'))
file_options = {f.name: f.name for f in json_files} json_files = [f for f in json_files if f.name not in (
'.editor_config.json', '.editor_snippets.json')]
file_options = {f.name: f.name for f in json_files}
src_file_select = ui.select( src_file_select = ui.select(
file_options, file_options,
value=file_path.name, value=file_path.name,
label='Source File:', label='Source File:',
).classes('w-64') ).classes('w-64')
src_seq_select = ui.select([], label='Source Sequence:').classes('w-64') src_seq_select = ui.select([], label='Source Sequence:').classes('w-64')
# Track loaded source data # Track loaded source data
_src_cache = {'data': None, 'batch': [], 'name': None} _src_cache = {'data': None, 'batch': [], 'name': None}
def _update_src(): def _update_src():
name = src_file_select.value name = src_file_select.value
if name and name != _src_cache['name']: if name and name != _src_cache['name']:
src_data, _ = load_json(state.current_dir / name) src_data, _ = load_json(state.current_dir / name)
_src_cache['data'] = src_data _src_cache['data'] = src_data
_src_cache['batch'] = src_data.get(KEY_BATCH_DATA, []) _src_cache['batch'] = src_data.get(KEY_BATCH_DATA, [])
_src_cache['name'] = name _src_cache['name'] = name
if _src_cache['batch']: if _src_cache['batch']:
opts = {i: format_seq_label(s.get(KEY_SEQUENCE_NUMBER, i+1)) opts = {i: format_seq_label(s.get(KEY_SEQUENCE_NUMBER, i+1))
for i, s in enumerate(_src_cache['batch'])} for i, s in enumerate(_src_cache['batch'])}
src_seq_select.set_options(opts, value=0) src_seq_select.set_options(opts, value=0)
else: else:
src_seq_select.set_options({}) src_seq_select.set_options({})
src_file_select.on_value_change(lambda _: _update_src()) src_file_select.on_value_change(lambda _: _update_src())
_update_src() _update_src()
# --- Add New Sequence --- def _add_sequence(new_item):
ui.label('Add New Sequence').classes('text-subtitle1 q-mt-md') new_item[KEY_SEQUENCE_NUMBER] = max_main_seq_number(batch_list) + 1
for k in [KEY_PROMPT_HISTORY, KEY_HISTORY_TREE, 'note', 'loras']:
new_item.pop(k, None)
batch_list.append(new_item)
data[KEY_BATCH_DATA] = batch_list
save_json(file_path, data)
render_sequence_list.refresh()
def _add_sequence(new_item): with ui.row().classes('q-mt-sm'):
new_item[KEY_SEQUENCE_NUMBER] = max_main_seq_number(batch_list) + 1 def add_empty():
for k in [KEY_PROMPT_HISTORY, KEY_HISTORY_TREE, 'note', 'loras']: _add_sequence(DEFAULTS.copy())
new_item.pop(k, None)
batch_list.append(new_item)
data[KEY_BATCH_DATA] = batch_list
save_json(file_path, data)
render_sequence_list.refresh()
with ui.row(): def add_from_source():
def add_empty(): item = copy.deepcopy(DEFAULTS)
_add_sequence(DEFAULTS.copy()) src_batch = _src_cache['batch']
sel_idx = src_seq_select.value
if src_batch and sel_idx is not None:
item.update(copy.deepcopy(src_batch[int(sel_idx)]))
elif _src_cache['data']:
item.update(copy.deepcopy(_src_cache['data']))
_add_sequence(item)
def add_from_source(): ui.button('Add Empty', icon='add', on_click=add_empty)
item = copy.deepcopy(DEFAULTS) ui.button('From Source', icon='file_download', on_click=add_from_source)
src_batch = _src_cache['batch']
sel_idx = src_seq_select.value
if src_batch and sel_idx is not None:
item.update(copy.deepcopy(src_batch[int(sel_idx)]))
elif _src_cache['data']:
item.update(copy.deepcopy(_src_cache['data']))
_add_sequence(item)
ui.button('Add Empty', icon='add', on_click=add_empty)
ui.button('From Source', icon='file_download', on_click=add_from_source)
ui.separator()
# --- Standard / LoRA / VACE key sets --- # --- Standard / LoRA / VACE key sets ---
lora_keys = ['lora 1 high', 'lora 1 low', 'lora 2 high', 'lora 2 low', lora_keys = ['lora 1 high', 'lora 1 low', 'lora 2 high', 'lora 2 low',
@@ -250,36 +264,36 @@ def render_batch_processor(state: AppState):
ui.button('Sort by Number', icon='sort', on_click=sort_by_number).props('flat') ui.button('Sort by Number', icon='sort', on_click=sort_by_number).props('flat')
for i, seq in enumerate(batch_list): for i, seq in enumerate(batch_list):
_render_sequence_card( with ui.card().classes('w-full q-mb-sm'):
i, seq, batch_list, data, file_path, state, _render_sequence_card(
_src_cache, src_seq_select, i, seq, batch_list, data, file_path, state,
standard_keys, render_sequence_list, _src_cache, src_seq_select,
) standard_keys, render_sequence_list,
)
render_sequence_list() render_sequence_list()
ui.separator()
# --- Save & Snap --- # --- Save & Snap ---
with ui.row().classes('w-full items-end q-gutter-md'): with ui.card().classes('w-full q-pa-md q-mt-lg'):
commit_input = ui.input('Change Note (Optional)', with ui.row().classes('w-full items-end q-gutter-md'):
placeholder='e.g. Added sequence 3').classes('col') commit_input = ui.input('Change Note (Optional)',
placeholder='e.g. Added sequence 3').classes('col')
def save_and_snap(): def save_and_snap():
data[KEY_BATCH_DATA] = batch_list data[KEY_BATCH_DATA] = batch_list
tree_data = data.get(KEY_HISTORY_TREE, {}) tree_data = data.get(KEY_HISTORY_TREE, {})
htree = HistoryTree(tree_data) htree = HistoryTree(tree_data)
snapshot_payload = copy.deepcopy(data) snapshot_payload = copy.deepcopy(data)
snapshot_payload.pop(KEY_HISTORY_TREE, None) snapshot_payload.pop(KEY_HISTORY_TREE, None)
note = commit_input.value if commit_input.value else 'Batch Update' note = commit_input.value if commit_input.value else 'Batch Update'
htree.commit(snapshot_payload, note=note) htree.commit(snapshot_payload, note=note)
data[KEY_HISTORY_TREE] = htree.to_dict() data[KEY_HISTORY_TREE] = htree.to_dict()
save_json(file_path, data) save_json(file_path, data)
state.restored_indicator = None state.restored_indicator = None
commit_input.set_value('') commit_input.set_value('')
ui.notify('Batch Saved & Snapshot Created!', type='positive') ui.notify('Batch Saved & Snapshot Created!', type='positive')
ui.button('Save & Snap', icon='save', on_click=save_and_snap).props('color=primary') ui.button('Save & Snap', icon='save', on_click=save_and_snap).props('color=primary')
# ====================================================================== # ======================================================================
@@ -321,7 +335,7 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
batch_list[idx] = item batch_list[idx] = item
commit('Copied!') commit('Copied!')
ui.button('Copy Src', icon='file_download', on_click=copy_source).props('dense') ui.button('Copy Src', icon='file_download', on_click=copy_source).props('outline')
# Clone Next # Clone Next
def clone_next(idx=i, sn=seq_num, s=seq): def clone_next(idx=i, sn=seq_num, s=seq):
@@ -334,7 +348,7 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
batch_list.insert(pos, new_seq) batch_list.insert(pos, new_seq)
commit('Cloned to Next!') commit('Cloned to Next!')
ui.button('Clone Next', icon='content_copy', on_click=clone_next).props('dense') ui.button('Clone Next', icon='content_copy', on_click=clone_next).props('outline')
# Clone End # Clone End
def clone_end(s=seq): def clone_end(s=seq):
@@ -343,7 +357,7 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
batch_list.append(new_seq) batch_list.append(new_seq)
commit('Cloned to End!') commit('Cloned to End!')
ui.button('Clone End', icon='vertical_align_bottom', on_click=clone_end).props('dense') ui.button('Clone End', icon='vertical_align_bottom', on_click=clone_end).props('outline')
# Clone Sub # Clone Sub
def clone_sub(idx=i, sn=seq_num, s=seq): def clone_sub(idx=i, sn=seq_num, s=seq):
@@ -360,29 +374,16 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
batch_list.insert(pos, new_seq) batch_list.insert(pos, new_seq)
commit(f'Created {format_seq_label(new_seq[KEY_SEQUENCE_NUMBER])}!') commit(f'Created {format_seq_label(new_seq[KEY_SEQUENCE_NUMBER])}!')
ui.button('Clone Sub', icon='link', on_click=clone_sub).props('dense') ui.button('Clone Sub', icon='link', on_click=clone_sub).props('outline')
# Promote ui.element('div').classes('col')
def promote(idx=i, s=seq):
single_data = copy.deepcopy(s)
single_data[KEY_PROMPT_HISTORY] = copy.deepcopy(data.get(KEY_PROMPT_HISTORY, []))
single_data[KEY_HISTORY_TREE] = copy.deepcopy(data.get(KEY_HISTORY_TREE, {}))
single_data.pop(KEY_SEQUENCE_NUMBER, None)
save_json(file_path, single_data)
state.data_cache = single_data
ui.notify('Converted to Single!', type='positive')
# Full refresh so batch tab re-enters render_batch_processor
# and sees the file is now single (no KEY_BATCH_DATA)
state._render_main.refresh()
ui.button('Promote', icon='north_west', on_click=promote).props('dense')
# Delete # Delete
def delete(idx=i): def delete(idx=i):
batch_list.pop(idx) batch_list.pop(idx)
commit() commit()
ui.button(icon='delete', on_click=delete).props('dense color=negative') ui.button(icon='delete', on_click=delete).props('color=negative')
ui.separator() ui.separator()
@@ -390,13 +391,13 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
with ui.splitter(value=66).classes('w-full') as splitter: with ui.splitter(value=66).classes('w-full') as splitter:
with splitter.before: with splitter.before:
dict_textarea('General Prompt', seq, 'general_prompt').classes( dict_textarea('General Prompt', seq, 'general_prompt').classes(
'w-full').props('outlined rows=2') 'w-full q-mt-sm').props('outlined rows=2')
dict_textarea('General Negative', seq, 'general_negative').classes( dict_textarea('General Negative', seq, 'general_negative').classes(
'w-full').props('outlined rows=2') 'w-full q-mt-sm').props('outlined rows=2')
dict_textarea('Specific Prompt', seq, 'current_prompt').classes( dict_textarea('Specific Prompt', seq, 'current_prompt').classes(
'w-full').props('outlined rows=10') 'w-full q-mt-sm').props('outlined rows=10')
dict_textarea('Specific Negative', seq, 'negative').classes( dict_textarea('Specific Negative', seq, 'negative').classes(
'w-full').props('outlined rows=2') 'w-full q-mt-sm').props('outlined rows=2')
with splitter.after: with splitter.after:
# Sequence number # Sequence number
@@ -426,7 +427,7 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
dict_input(ui.input, 'FLF', seq, 'flf').props('outlined').classes('w-full') dict_input(ui.input, 'FLF', seq, 'flf').props('outlined').classes('w-full')
dict_number('End Frame', seq, 'end_frame').props('outlined').classes('w-full') dict_number('End Frame', seq, 'end_frame').props('outlined').classes('w-full')
dict_input(ui.input, 'Video File Path', seq, 'video file path').props( dict_input(ui.input, 'Video File Path', seq, 'video file path').props(
'outlined').classes('w-full') 'outlined input-style="direction: rtl"').classes('w-full')
# Image paths with preview # Image paths with preview
for img_label, img_key in [ for img_label, img_key in [
@@ -436,7 +437,7 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
]: ]:
with ui.row().classes('w-full items-center'): with ui.row().classes('w-full items-center'):
inp = dict_input(ui.input, img_label, seq, img_key).classes( inp = dict_input(ui.input, img_label, seq, img_key).classes(
'col').props('outlined') 'col').props('outlined input-style="direction: rtl"')
img_path = Path(seq.get(img_key, '')) if seq.get(img_key) else None img_path = Path(seq.get(img_key, '')) if seq.get(img_key) else None
if (img_path and img_path.exists() and if (img_path and img_path.exists() and
img_path.suffix.lower() in IMAGE_EXTENSIONS): img_path.suffix.lower() in IMAGE_EXTENSIONS):
@@ -444,38 +445,56 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
ui.image(str(img_path)).classes('w-full') ui.image(str(img_path)).classes('w-full')
ui.button(icon='visibility', on_click=dlg.open).props('flat dense') ui.button(icon='visibility', on_click=dlg.open).props('flat dense')
# VACE Settings # --- VACE Settings (full width) ---
with ui.expansion('VACE Settings', icon='settings').classes('w-full'): with ui.expansion('VACE Settings', icon='settings').classes('w-full'):
_render_vace_settings(i, seq, batch_list, data, file_path, refresh_list) _render_vace_settings(i, seq, batch_list, data, file_path, refresh_list)
# --- LoRA Settings --- # --- LoRA Settings ---
with ui.expansion('LoRA Settings', icon='style').classes('w-full'): with ui.expansion('LoRA Settings', icon='style').classes('w-full'):
with ui.row().classes('w-full q-gutter-md'): for lora_idx in range(1, 4):
for lora_idx in range(1, 4): for tier, tier_label in [('high', 'High'), ('low', 'Low')]:
with ui.column().classes('col'): k = f'lora {lora_idx} {tier}'
ui.label(f'LoRA {lora_idx}').classes('text-subtitle2') raw = str(seq.get(k, ''))
for tier, tier_label in [('high', 'High'), ('low', 'Low')]: inner = raw.replace('<lora:', '').replace('>', '')
k = f'lora {lora_idx} {tier}' # Split "name:strength" or just "name"
raw = str(seq.get(k, '')) if ':' in inner:
disp = raw.replace('<lora:', '').replace('>', '') parts = inner.rsplit(':', 1)
lora_name = parts[0]
try:
lora_strength = float(parts[1])
except ValueError:
lora_name = inner
lora_strength = 1.0
else:
lora_name = inner
lora_strength = 1.0
with ui.row().classes('w-full items-center'): with ui.row().classes('w-full items-center q-gutter-sm'):
ui.label('<lora:').classes('text-caption font-mono') ui.label(f'L{lora_idx} {tier_label}').classes(
lora_input = ui.input( 'text-caption').style('min-width: 55px')
f'L{lora_idx} {tier_label}', name_input = ui.input(
value=disp, 'Name',
).classes('col').props('outlined dense') value=lora_name,
ui.label('>').classes('text-caption font-mono') ).classes('col').props('outlined dense')
strength_input = ui.number(
'Str',
value=lora_strength,
min=0, max=10, step=0.1,
format='%.1f',
).props('outlined dense').style('max-width: 80px')
def on_lora_blur(e, key=k): def _lora_sync(key=k, n_inp=name_input, s_inp=strength_input):
v = e.sender.value name = n_inp.value or ''
seq[key] = f'<lora:{v}>' if v else '' strength = s_inp.value if s_inp.value is not None else 1.0
seq[key] = f'<lora:{name}:{strength:.1f}>' if name else ''
lora_input.on('blur', on_lora_blur) name_input.on('blur', lambda _, s=_lora_sync: s())
name_input.on('update:model-value', lambda _, s=_lora_sync: s())
strength_input.on('blur', lambda _, s=_lora_sync: s())
strength_input.on('update:model-value', lambda _, s=_lora_sync: s())
# --- Custom Parameters --- # --- Custom Parameters ---
ui.separator() ui.label('Custom Parameters').classes('section-header q-mt-md')
ui.label('Custom Parameters').classes('text-caption')
custom_keys = [k for k in seq.keys() if k not in standard_keys] custom_keys = [k for k in seq.keys() if k not in standard_keys]
if custom_keys: if custom_keys:
@@ -511,49 +530,10 @@ def _render_sequence_card(i, seq, batch_list, data, file_path, state,
# ====================================================================== # ======================================================================
def _render_vace_settings(i, seq, batch_list, data, file_path, refresh_list): def _render_vace_settings(i, seq, batch_list, data, file_path, refresh_list):
# Frame to Skip + shift # VACE Schedule (needed early for both columns)
with ui.row().classes('w-full items-end'):
fts_input = dict_number('Frame to Skip', seq, 'frame_to_skip').classes('col').props(
'outlined')
# Capture original at render time; blur updates seq before click fires
_original_fts = int(seq.get('frame_to_skip', FRAME_TO_SKIP_DEFAULT))
def shift_fts(idx=i, orig=_original_fts):
new_fts = int(fts_input.value) if fts_input.value is not None else orig
delta = new_fts - orig
if delta == 0:
ui.notify('No change to shift', type='info')
return
shifted = 0
for j in range(idx + 1, len(batch_list)):
batch_list[j]['frame_to_skip'] = int(
batch_list[j].get('frame_to_skip', FRAME_TO_SKIP_DEFAULT)) + delta
shifted += 1
data[KEY_BATCH_DATA] = batch_list
save_json(file_path, data)
ui.notify(f'Shifted {shifted} sequences by {delta:+d}', type='positive')
refresh_list.refresh()
ui.button('Shift', icon='arrow_downward', on_click=shift_fts).props('dense')
dict_input(ui.input, 'Transition', seq, 'transition').props('outlined')
# VACE Schedule
sched_val = max(0, min(int(seq.get('vace schedule', 1)), len(VACE_MODES) - 1)) sched_val = max(0, min(int(seq.get('vace schedule', 1)), len(VACE_MODES) - 1))
with ui.row().classes('w-full items-center'):
vs_input = dict_number('VACE Schedule', seq, 'vace schedule',
min=0, max=len(VACE_MODES) - 1).classes('col').props('outlined')
mode_label = ui.label(VACE_MODES[sched_val]).classes('text-caption')
def update_mode_label(e): # Mode reference dialog
idx = int(e.sender.value) if e.sender.value is not None else 0
idx = max(0, min(idx, len(VACE_MODES) - 1))
mode_label.set_text(VACE_MODES[idx])
vs_input.on('update:model-value', update_mode_label)
# Mode reference
with ui.dialog() as ref_dlg, ui.card(): with ui.dialog() as ref_dlg, ui.card():
table_md = ( table_md = (
'| # | Mode | Formula |\n|:--|:-----|:--------|\n' '| # | Mode | Formula |\n|:--|:-----|:--------|\n'
@@ -563,30 +543,83 @@ def _render_vace_settings(i, seq, batch_list, data, file_path, refresh_list):
+ '\n\n*All totals snapped to 4n+1 (1,5,9,...,49,...,81,...)*' + '\n\n*All totals snapped to 4n+1 (1,5,9,...,49,...,81,...)*'
) )
ui.markdown(table_md) ui.markdown(table_md)
ui.button('Mode Reference', icon='help', on_click=ref_dlg.open).props('flat dense')
# Input A / B frames with ui.row().classes('w-full q-gutter-md'):
ia_input = dict_number('Input A Frames', seq, 'input_a_frames').props('outlined') # --- Left column ---
ib_input = dict_number('Input B Frames', seq, 'input_b_frames').props('outlined') with ui.column().classes('col'):
# Frame to Skip + shift
with ui.row().classes('w-full items-end'):
fts_input = dict_number('Frame to Skip', seq, 'frame_to_skip').classes(
'col').props('outlined')
# VACE Length + output calculation _original_fts = int(seq.get('frame_to_skip', FRAME_TO_SKIP_DEFAULT))
input_a = int(seq.get('input_a_frames', 16))
input_b = int(seq.get('input_b_frames', 16))
stored_total = int(seq.get('vace_length', 49))
mode_idx = int(seq.get('vace schedule', 1))
if mode_idx == 0: def shift_fts(idx=i, orig=_original_fts):
base_length = max(stored_total - input_a, 1) new_fts = int(fts_input.value) if fts_input.value is not None else orig
elif mode_idx == 1: delta = new_fts - orig
base_length = max(stored_total - input_b, 1) if delta == 0:
else: ui.notify('No change to shift', type='info')
base_length = max(stored_total - input_a - input_b, 1) return
shifted = 0
for j in range(idx + 1, len(batch_list)):
batch_list[j]['frame_to_skip'] = int(
batch_list[j].get('frame_to_skip', FRAME_TO_SKIP_DEFAULT)) + delta
shifted += 1
data[KEY_BATCH_DATA] = batch_list
save_json(file_path, data)
ui.notify(f'Shifted {shifted} sequences by {delta:+d}', type='positive')
refresh_list.refresh()
with ui.row().classes('w-full items-center'): ui.button('Shift', icon='arrow_downward', on_click=shift_fts).props(
vl_input = ui.number('VACE Length', value=base_length, min=1).classes('col').props( 'outline').style('height: 40px')
'outlined')
output_label = ui.label(f'Output: {stored_total}').classes('text-bold')
dict_input(ui.input, 'Transition', seq, 'transition').props('outlined').classes(
'w-full q-mt-sm')
# VACE Schedule
with ui.row().classes('w-full items-center q-mt-sm'):
vs_input = dict_number('VACE Schedule', seq, 'vace schedule', default=1,
min=0, max=len(VACE_MODES) - 1).classes('col').props(
'outlined')
mode_label = ui.label(VACE_MODES[sched_val]).classes('text-caption')
ui.button(icon='help', on_click=ref_dlg.open).props('flat dense round')
def update_mode_label(e):
idx = int(e.sender.value) if e.sender.value is not None else 0
idx = max(0, min(idx, len(VACE_MODES) - 1))
mode_label.set_text(VACE_MODES[idx])
vs_input.on('update:model-value', update_mode_label)
# --- Right column ---
with ui.column().classes('col'):
ia_input = dict_number('Input A Frames', seq, 'input_a_frames').props(
'outlined').classes('w-full')
ib_input = dict_number('Input B Frames', seq, 'input_b_frames').props(
'outlined').classes('w-full q-mt-sm')
# VACE Length + output calculation
input_a = int(seq.get('input_a_frames', 16))
input_b = int(seq.get('input_b_frames', 16))
stored_total = int(seq.get('vace_length', 49))
mode_idx = int(seq.get('vace schedule', 1))
if mode_idx == 0:
base_length = max(stored_total - input_a, 1)
elif mode_idx == 1:
base_length = max(stored_total - input_b, 1)
else:
base_length = max(stored_total - input_a - input_b, 1)
with ui.row().classes('w-full items-center q-mt-sm'):
vl_input = ui.number('VACE Length', value=base_length, min=1).classes(
'col').props('outlined')
output_label = ui.label(f'Output: {stored_total}').classes('text-bold')
dict_number('Reference Switch', seq, 'reference switch').props(
'outlined').classes('w-full q-mt-sm')
# Recalculate VACE output when any input changes
def recalc_vace(*_args): def recalc_vace(*_args):
mi = int(vs_input.value) if vs_input.value is not None else 0 mi = int(vs_input.value) if vs_input.value is not None else 0
ia = int(ia_input.value) if ia_input.value is not None else 16 ia = int(ia_input.value) if ia_input.value is not None else 16
@@ -607,8 +640,6 @@ def _render_vace_settings(i, seq, batch_list, data, file_path, refresh_list):
for inp in (vs_input, ia_input, ib_input, vl_input): for inp in (vs_input, ia_input, ib_input, vl_input):
inp.on('update:model-value', recalc_vace) inp.on('update:model-value', recalc_vace)
dict_number('Reference Switch', seq, 'reference switch').props('outlined')
# ====================================================================== # ======================================================================
# Mass Update # Mass Update
@@ -638,13 +669,14 @@ def _render_mass_update(batch_list, data, file_path, state: AppState, refresh_li
source_select.on_value_change(update_fields) source_select.on_value_change(update_fields)
update_fields() update_fields()
ui.label('Apply to:').classes('text-subtitle2 q-mt-md') ui.label('Apply to:').classes('subsection-header q-mt-md')
select_all_cb = ui.checkbox('Select All') select_all_cb = ui.checkbox('Select All')
target_checks = {} target_checks = {}
for idx, s in enumerate(batch_list): with ui.scroll_area().style('max-height: 250px'):
sn = s.get(KEY_SEQUENCE_NUMBER, idx + 1) for idx, s in enumerate(batch_list):
cb = ui.checkbox(format_seq_label(sn)) sn = s.get(KEY_SEQUENCE_NUMBER, idx + 1)
target_checks[idx] = cb cb = ui.checkbox(format_seq_label(sn))
target_checks[idx] = cb
def on_select_all(e): def on_select_all(e):
for cb in target_checks.values(): for cb in target_checks.values():

View File

@@ -55,7 +55,7 @@ def render_comfy_monitor(state: AppState):
# Add server section # Add server section
ui.separator() ui.separator()
ui.label('Add New Server').classes('text-subtitle1') ui.label('Add New Server').classes('section-header')
with ui.row().classes('w-full items-end'): with ui.row().classes('w-full items-end'):
new_name = ui.input('Server Name', placeholder='e.g. Render Node 2').classes('col') new_name = ui.input('Server Name', placeholder='e.g. Render Node 2').classes('col')
new_url = ui.input('URL', placeholder='http://192.168.1.50:8188').classes('col') new_url = ui.input('URL', placeholder='http://192.168.1.50:8188').classes('col')
@@ -152,18 +152,18 @@ def _render_single_instance(state: AppState, instance_config: dict, index: int,
running_cnt = len(queue_data.get('queue_running', [])) running_cnt = len(queue_data.get('queue_running', []))
pending_cnt = len(queue_data.get('queue_pending', [])) pending_cnt = len(queue_data.get('queue_pending', []))
with ui.card().classes('q-pa-sm'): with ui.card().classes('q-pa-md text-center').style('min-width: 100px'):
ui.label('Status') ui.label('Status')
ui.label('Online' if running_cnt > 0 else 'Idle').classes( ui.label('Online' if running_cnt > 0 else 'Idle').classes(
'text-positive' if running_cnt > 0 else 'text-grey') 'text-positive' if running_cnt > 0 else 'text-grey')
with ui.card().classes('q-pa-sm'): with ui.card().classes('q-pa-md text-center').style('min-width: 100px'):
ui.label('Pending') ui.label('Pending')
ui.label(str(pending_cnt)) ui.label(str(pending_cnt))
with ui.card().classes('q-pa-sm'): with ui.card().classes('q-pa-md text-center').style('min-width: 100px'):
ui.label('Running') ui.label('Running')
ui.label(str(running_cnt)) ui.label(str(running_cnt))
else: else:
with ui.card().classes('q-pa-sm'): with ui.card().classes('q-pa-md text-center').style('min-width: 100px'):
ui.label('Status') ui.label('Status')
ui.label('Offline').classes('text-negative') ui.label('Offline').classes('text-negative')
ui.label(f'Could not connect to {comfy_url}').classes('text-negative') ui.label(f'Could not connect to {comfy_url}').classes('text-negative')
@@ -173,104 +173,106 @@ def _render_single_instance(state: AppState, instance_config: dict, index: int,
ui.button('Refresh Status', icon='refresh', on_click=refresh_status).props('flat dense') ui.button('Refresh Status', icon='refresh', on_click=refresh_status).props('flat dense')
# --- Live View --- # --- Live View ---
ui.label('Live View').classes('text-subtitle1 q-mt-md') with ui.card().classes('w-full q-pa-md q-mt-md'):
toggle_key = f'live_toggle_{index}' ui.label('Live View').classes('section-header')
toggle_key = f'live_toggle_{index}'
live_checkbox = ui.checkbox('Enable Live Preview', value=False) live_checkbox = ui.checkbox('Enable Live Preview', value=False)
# Store reference so poll_all timer can disable it on timeout # Store reference so poll_all timer can disable it on timeout
state._live_checkboxes[toggle_key] = live_checkbox state._live_checkboxes[toggle_key] = live_checkbox
@ui.refreshable @ui.refreshable
def render_live_view(): def render_live_view():
if not live_checkbox.value: if not live_checkbox.value:
ui.label('Live Preview is disabled.').classes('text-caption') ui.label('Live Preview is disabled.').classes('text-caption')
return
# Record start time
if toggle_key not in state.live_toggles or state.live_toggles.get(toggle_key) is None:
state.live_toggles[toggle_key] = time.time()
timeout_val = config.get('monitor_timeout', 0)
if timeout_val > 0:
start = state.live_toggles.get(toggle_key, time.time())
remaining = (timeout_val * 60) - (time.time() - start)
if remaining <= 0:
live_checkbox.set_value(False)
state.live_toggles[toggle_key] = None
ui.label('Preview timed out.').classes('text-caption')
return return
ui.label(f'Auto-off in: {int(remaining)}s').classes('text-caption')
iframe_h = ui.slider(min=600, max=2500, step=50, value=1000).classes('w-full') # Record start time
ui.label().bind_text_from(iframe_h, 'value', backward=lambda v: f'Height: {v}px') if toggle_key not in state.live_toggles or state.live_toggles.get(toggle_key) is None:
state.live_toggles[toggle_key] = time.time()
viewer_base = config.get('viewer_url', '').strip() timeout_val = config.get('monitor_timeout', 0)
parsed = urllib.parse.urlparse(viewer_base) if timeout_val > 0:
if viewer_base and parsed.scheme in ('http', 'https'): start = state.live_toggles.get(toggle_key, time.time())
safe_src = html.escape(viewer_base, quote=True) remaining = (timeout_val * 60) - (time.time() - start)
ui.label(f'Viewing: {viewer_base}').classes('text-caption') if remaining <= 0:
live_checkbox.set_value(False)
state.live_toggles[toggle_key] = None
ui.label('Preview timed out.').classes('text-caption')
return
ui.label(f'Auto-off in: {int(remaining)}s').classes('text-caption')
iframe_container = ui.column().classes('w-full') iframe_h = ui.slider(min=600, max=2500, step=50, value=1000).classes('w-full')
ui.label().bind_text_from(iframe_h, 'value', backward=lambda v: f'Height: {v}px')
def update_iframe(): viewer_base = config.get('viewer_url', '').strip()
iframe_container.clear() parsed = urllib.parse.urlparse(viewer_base)
with iframe_container: if viewer_base and parsed.scheme in ('http', 'https'):
ui.html( safe_src = html.escape(viewer_base, quote=True)
f'<iframe src="{safe_src}" width="100%" height="{int(iframe_h.value)}px"' ui.label(f'Viewing: {viewer_base}').classes('text-caption')
f' style="border: 2px solid #666; border-radius: 8px;"></iframe>'
)
iframe_h.on_value_change(lambda _: update_iframe()) iframe_container = ui.column().classes('w-full')
update_iframe()
else:
ui.label('No valid viewer URL configured.').classes('text-warning')
state._live_refreshables[toggle_key] = render_live_view def update_iframe():
live_checkbox.on_value_change(lambda _: render_live_view.refresh()) iframe_container.clear()
render_live_view() with iframe_container:
ui.html(
f'<iframe src="{safe_src}" width="100%" height="{int(iframe_h.value)}px"'
f' style="border: 2px solid #666; border-radius: 8px;"></iframe>'
)
iframe_h.on_value_change(lambda _: update_iframe())
update_iframe()
else:
ui.label('No valid viewer URL configured.').classes('text-warning')
state._live_refreshables[toggle_key] = render_live_view
live_checkbox.on_value_change(lambda _: render_live_view.refresh())
render_live_view()
# --- Latest Output --- # --- Latest Output ---
ui.label('Latest Output').classes('text-subtitle1 q-mt-md') with ui.card().classes('w-full q-pa-md q-mt-md'):
img_container = ui.column().classes('w-full') ui.label('Latest Output').classes('section-header')
img_container = ui.column().classes('w-full')
async def check_image(): async def check_image():
img_container.clear() img_container.clear()
loop = asyncio.get_event_loop() loop = asyncio.get_event_loop()
res, err = await loop.run_in_executor( res, err = await loop.run_in_executor(
None, lambda: _fetch_blocking(f'{comfy_url}/history', timeout=2)) None, lambda: _fetch_blocking(f'{comfy_url}/history', timeout=2))
with img_container: with img_container:
if err is not None: if err is not None:
ui.label(f'Error fetching image: {err}').classes('text-negative') ui.label(f'Error fetching image: {err}').classes('text-negative')
return return
try: try:
history = res.json() history = res.json()
except (ValueError, Exception): except (ValueError, Exception):
ui.label('Invalid response from server').classes('text-negative') ui.label('Invalid response from server').classes('text-negative')
return return
if not history: if not history:
ui.label('No history found.').classes('text-caption') ui.label('No history found.').classes('text-caption')
return return
last_prompt_id = list(history.keys())[-1] last_prompt_id = list(history.keys())[-1]
outputs = history[last_prompt_id].get('outputs', {}) outputs = history[last_prompt_id].get('outputs', {})
found_img = None found_img = None
for node_output in outputs.values(): for node_output in outputs.values():
if 'images' in node_output: if 'images' in node_output:
for img_info in node_output['images']: for img_info in node_output['images']:
if img_info['type'] == 'output': if img_info['type'] == 'output':
found_img = img_info found_img = img_info
break break
if found_img:
break
if found_img: if found_img:
break params = urllib.parse.urlencode({
if found_img: 'filename': found_img['filename'],
params = urllib.parse.urlencode({ 'subfolder': found_img['subfolder'],
'filename': found_img['filename'], 'type': found_img['type'],
'subfolder': found_img['subfolder'], })
'type': found_img['type'], img_url = f'{comfy_url}/view?{params}'
}) ui.image(img_url).classes('w-full').style('max-width: 600px')
img_url = f'{comfy_url}/view?{params}' ui.label(f'Last Output: {found_img["filename"]}').classes('text-caption')
ui.image(img_url).classes('w-full').style('max-width: 600px') else:
ui.label(f'Last Output: {found_img["filename"]}').classes('text-caption') ui.label('Last run had no image output.').classes('text-caption')
else:
ui.label('Last run had no image output.').classes('text-caption')
ui.button('Check Latest Image', icon='image', on_click=check_image).props('flat') ui.button('Check Latest Image', icon='image', on_click=check_image).props('flat')

View File

@@ -11,64 +11,63 @@ def render_raw_editor(state: AppState):
data = state.data_cache data = state.data_cache
file_path = state.file_path file_path = state.file_path
ui.label(f'Raw Editor: {file_path.name}').classes('text-h6') with ui.card().classes('w-full q-pa-md'):
ui.label(f'Raw Editor: {file_path.name}').classes('text-h6 q-mb-md')
hide_history = ui.checkbox( hide_history = ui.checkbox(
'Hide History (Safe Mode)', 'Hide History (Safe Mode)',
value=True, value=True,
) )
@ui.refreshable @ui.refreshable
def render_editor(): def render_editor():
# Prepare display data # Prepare display data
if hide_history.value: if hide_history.value:
display_data = copy.deepcopy(data) display_data = copy.deepcopy(data)
display_data.pop(KEY_HISTORY_TREE, None) display_data.pop(KEY_HISTORY_TREE, None)
display_data.pop(KEY_PROMPT_HISTORY, None) display_data.pop(KEY_PROMPT_HISTORY, None)
else: else:
display_data = data display_data = data
try:
json_str = json.dumps(display_data, indent=4, ensure_ascii=False)
except Exception as e:
ui.notify(f'Error serializing JSON: {e}', type='negative')
json_str = '{}'
text_area = ui.textarea(
'JSON Content',
value=json_str,
).classes('w-full font-mono').props('outlined rows=30')
ui.separator()
def do_save():
try: try:
input_data = json.loads(text_area.value) json_str = json.dumps(display_data, indent=4, ensure_ascii=False)
# Merge hidden history back in if safe mode
if hide_history.value:
if KEY_HISTORY_TREE in data:
input_data[KEY_HISTORY_TREE] = data[KEY_HISTORY_TREE]
if KEY_PROMPT_HISTORY in data:
input_data[KEY_PROMPT_HISTORY] = data[KEY_PROMPT_HISTORY]
save_json(file_path, input_data)
data.clear()
data.update(input_data)
state.last_mtime = get_file_mtime(file_path)
ui.notify('Raw JSON Saved Successfully!', type='positive')
render_editor.refresh()
except json.JSONDecodeError as e:
ui.notify(f'Invalid JSON Syntax: {e}', type='negative')
except Exception as e: except Exception as e:
ui.notify(f'Unexpected Error: {e}', type='negative') ui.notify(f'Error serializing JSON: {e}', type='negative')
json_str = '{}'
ui.button('Save Raw Changes', icon='save', on_click=do_save).props( text_area = ui.textarea(
'color=primary' 'JSON Content',
).classes('w-full') value=json_str,
).classes('w-full font-mono').props('outlined rows=30')
hide_history.on_value_change(lambda _: render_editor.refresh()) def do_save():
render_editor() try:
input_data = json.loads(text_area.value)
# Merge hidden history back in if safe mode
if hide_history.value:
if KEY_HISTORY_TREE in data:
input_data[KEY_HISTORY_TREE] = data[KEY_HISTORY_TREE]
if KEY_PROMPT_HISTORY in data:
input_data[KEY_PROMPT_HISTORY] = data[KEY_PROMPT_HISTORY]
save_json(file_path, input_data)
data.clear()
data.update(input_data)
state.last_mtime = get_file_mtime(file_path)
ui.notify('Raw JSON Saved Successfully!', type='positive')
render_editor.refresh()
except json.JSONDecodeError as e:
ui.notify(f'Invalid JSON Syntax: {e}', type='negative')
except Exception as e:
ui.notify(f'Unexpected Error: {e}', type='negative')
ui.button('Save Raw Changes', icon='save', on_click=do_save).props(
'color=primary'
).classes('w-full q-mt-md')
hide_history.on_value_change(lambda _: render_editor.refresh())
render_editor()

View File

@@ -68,11 +68,12 @@ def _render_graph_or_log(mode, all_nodes, htree, selected_nodes,
"""Render graph visualization or linear log view.""" """Render graph visualization or linear log view."""
if mode in ('Horizontal', 'Vertical'): if mode in ('Horizontal', 'Vertical'):
direction = 'LR' if mode == 'Horizontal' else 'TB' direction = 'LR' if mode == 'Horizontal' else 'TB'
try: with ui.card().classes('w-full q-pa-md'):
graph_dot = htree.generate_graph(direction=direction) try:
_render_graphviz(graph_dot) graph_dot = htree.generate_graph(direction=direction)
except Exception as e: _render_graphviz(graph_dot)
ui.label(f'Graph Error: {e}').classes('text-negative') except Exception as e:
ui.label(f'Graph Error: {e}').classes('text-negative')
elif mode == 'Linear Log': elif mode == 'Linear Log':
ui.label('Chronological list of all snapshots.').classes('text-caption') ui.label('Chronological list of all snapshots.').classes('text-caption')
@@ -82,9 +83,9 @@ def _render_graph_or_log(mode, all_nodes, htree, selected_nodes,
card_style = '' card_style = ''
if is_selected: if is_selected:
card_style = 'background: #3d1f1f !important;' card_style = 'background: rgba(239, 68, 68, 0.1) !important; border-left: 3px solid var(--negative);'
elif is_head: elif is_head:
card_style = 'background: #1a2332 !important;' card_style = 'background: var(--accent-subtle) !important; border-left: 3px solid var(--accent);'
with ui.card().classes('w-full q-mb-sm').style(card_style): with ui.card().classes('w-full q-mb-sm').style(card_style):
with ui.row().classes('w-full items-center'): with ui.row().classes('w-full items-center'):
if selection_mode_on: if selection_mode_on:
@@ -129,10 +130,11 @@ def _render_batch_delete(htree, data, file_path, state, refresh_fn):
).classes('text-warning q-mt-md') ).classes('text-warning q-mt-md')
def do_batch_delete(): def do_batch_delete():
_delete_nodes(htree, data, file_path, valid) current_valid = state.timeline_selected_nodes & set(htree.nodes.keys())
_delete_nodes(htree, data, file_path, current_valid)
state.timeline_selected_nodes = set() state.timeline_selected_nodes = set()
ui.notify( ui.notify(
f'Deleted {count} node{"s" if count != 1 else ""}!', f'Deleted {len(current_valid)} node{"s" if len(current_valid) != 1 else ""}!',
type='positive') type='positive')
refresh_fn() refresh_fn()
@@ -143,70 +145,178 @@ def _render_batch_delete(htree, data, file_path, state, refresh_fn):
).props('color=negative') ).props('color=negative')
def _walk_branch_nodes(htree, tip_id):
"""Walk parent pointers from tip, returning nodes newest-first."""
nodes = []
current = tip_id
while current and current in htree.nodes:
nodes.append(htree.nodes[current])
current = htree.nodes[current].get('parent')
return nodes
def _find_active_branch(htree):
"""Return branch name whose tip == head_id, or None if detached."""
if not htree.head_id:
return None
for b_name, tip_id in htree.branches.items():
if tip_id == htree.head_id:
return b_name
return None
def _render_node_manager(all_nodes, htree, data, file_path, restore_fn, refresh_fn): def _render_node_manager(all_nodes, htree, data, file_path, restore_fn, refresh_fn):
"""Render node selector with restore, rename, delete, and preview.""" """Render branch-grouped node manager with restore, rename, delete, and preview."""
ui.label('Manage Version').classes('text-subtitle1 q-mt-md') ui.label('Manage Version').classes('section-header')
def fmt_node(n): # --- State that survives @ui.refreshable ---
ts = time.strftime('%b %d %H:%M', time.localtime(n['timestamp'])) active_branch = _find_active_branch(htree)
return f'{n.get("note", "Step")} - {ts} ({n["id"][:6]})'
node_options = {n['id']: fmt_node(n) for n in all_nodes} # Default branch: active branch, or branch whose ancestry contains HEAD
current_id = htree.head_id if htree.head_id in node_options else ( default_branch = active_branch
all_nodes[0]['id'] if all_nodes else None) if not default_branch and htree.head_id:
for b_name, tip_id in htree.branches.items():
for n in _walk_branch_nodes(htree, tip_id):
if n['id'] == htree.head_id:
default_branch = b_name
break
if default_branch:
break
if not default_branch and htree.branches:
default_branch = next(iter(htree.branches))
selected_node_id = ui.select( selected = {'node_id': htree.head_id, 'branch': default_branch}
node_options,
value=current_id, # --- (a) Branch selector ---
label='Select Version to Manage:', def fmt_branch(b_name):
count = len(_walk_branch_nodes(htree, htree.branches.get(b_name)))
suffix = ' (active)' if b_name == active_branch else ''
return f'{b_name} ({count} nodes){suffix}'
branch_options = {b: fmt_branch(b) for b in htree.branches}
def on_branch_change(e):
selected['branch'] = e.value
tip = htree.branches.get(e.value)
if tip:
selected['node_id'] = tip
render_branch_nodes.refresh()
ui.select(
branch_options,
value=selected['branch'],
label='Branch:',
on_change=on_branch_change,
).classes('w-full') ).classes('w-full')
with ui.row().classes('w-full items-end q-gutter-md'): # --- (b) Node list + (c) Actions panel ---
def restore_selected(): @ui.refreshable
nid = selected_node_id.value def render_branch_nodes():
if nid and nid in htree.nodes: branch_name = selected['branch']
restore_fn(htree.nodes[nid]) tip_id = htree.branches.get(branch_name)
nodes = _walk_branch_nodes(htree, tip_id) if tip_id else []
ui.button('Restore Version', icon='restore', if not nodes:
on_click=restore_selected).props('color=primary') ui.label('No nodes on this branch.').classes('text-caption q-pa-sm')
return
# Rename with ui.scroll_area().classes('w-full').style('max-height: 350px'):
with ui.row().classes('w-full items-end q-gutter-md'): for n in nodes:
rename_input = ui.input('Rename Label').classes('col') nid = n['id']
is_head = nid == htree.head_id
is_tip = nid == tip_id
is_selected = nid == selected['node_id']
def rename_node(): card_style = ''
nid = selected_node_id.value if is_selected:
if nid and nid in htree.nodes and rename_input.value: card_style = 'border-left: 3px solid var(--primary);'
htree.nodes[nid]['note'] = rename_input.value elif is_head:
data[KEY_HISTORY_TREE] = htree.to_dict() card_style = 'border-left: 3px solid var(--accent);'
save_json(file_path, data)
ui.notify('Label updated', type='positive')
refresh_fn()
ui.button('Update Label', on_click=rename_node).props('flat') with ui.card().classes('w-full q-mb-xs q-pa-xs').style(card_style):
with ui.row().classes('w-full items-center no-wrap'):
icon = 'location_on' if is_head else 'circle'
icon_size = 'sm' if is_head else 'xs'
ui.icon(icon, size=icon_size).classes(
'text-primary' if is_head else 'text-grey')
# Danger zone with ui.column().classes('col q-ml-xs').style('min-width: 0'):
with ui.expansion('Danger Zone (Delete)', icon='warning').classes('w-full q-mt-md'): note = n.get('note', 'Step')
ui.label('Deleting a node cannot be undone.').classes('text-warning') ts = time.strftime('%b %d %H:%M',
time.localtime(n['timestamp']))
label_text = note
lbl = ui.label(label_text).classes('text-body2 ellipsis')
if is_head:
lbl.classes('text-bold')
ui.label(f'{ts} \u2022 {nid[:6]}').classes(
'text-caption text-grey')
def delete_selected(): if is_head:
nid = selected_node_id.value ui.badge('HEAD', color='amber').props('dense')
if nid and nid in htree.nodes: if is_tip and not is_head:
_delete_nodes(htree, data, file_path, {nid}) ui.badge('tip', color='green', outline=True).props('dense')
ui.notify('Node Deleted', type='positive')
refresh_fn()
ui.button('Delete This Node', icon='delete', def select_node(node_id=nid):
on_click=delete_selected).props('color=negative') selected['node_id'] = node_id
render_branch_nodes.refresh()
# Data preview ui.button(icon='check_circle', on_click=select_node).props(
ui.separator() 'flat dense round size=sm'
with ui.expansion('Data Preview', icon='preview').classes('w-full'): ).tooltip('Select this node')
@ui.refreshable
def render_preview(): # --- (c) Actions panel ---
_render_data_preview(selected_node_id, htree) sel_id = selected['node_id']
selected_node_id.on_value_change(lambda _: render_preview.refresh()) if not sel_id or sel_id not in htree.nodes:
render_preview() return
sel_node = htree.nodes[sel_id]
sel_note = sel_node.get('note', 'Step')
is_head = sel_id == htree.head_id
ui.separator().classes('q-my-sm')
ui.label(f'Selected: {sel_note} ({sel_id[:6]})').classes(
'text-caption text-bold')
with ui.row().classes('w-full items-end q-gutter-sm'):
if not is_head:
def restore_selected():
if sel_id in htree.nodes:
restore_fn(htree.nodes[sel_id])
ui.button('Restore', icon='restore',
on_click=restore_selected).props('color=primary dense')
# Rename
rename_input = ui.input('Rename Label').classes('col').props('dense')
def rename_node():
if sel_id in htree.nodes and rename_input.value:
htree.nodes[sel_id]['note'] = rename_input.value
data[KEY_HISTORY_TREE] = htree.to_dict()
save_json(file_path, data)
ui.notify('Label updated', type='positive')
refresh_fn()
ui.button('Update Label', on_click=rename_node).props('flat dense')
# Danger zone
with ui.expansion('Danger Zone', icon='warning').classes(
'w-full q-mt-sm').style('border-left: 3px solid var(--negative)'):
ui.label('Deleting a node cannot be undone.').classes('text-warning')
def delete_selected():
if sel_id in htree.nodes:
_delete_nodes(htree, data, file_path, {sel_id})
ui.notify('Node Deleted', type='positive')
refresh_fn()
ui.button('Delete This Node', icon='delete',
on_click=delete_selected).props('color=negative dense')
# Data preview
with ui.expansion('Data Preview', icon='preview').classes('w-full q-mt-sm'):
_render_data_preview(sel_id, htree)
render_branch_nodes()
def render_timeline_tab(state: AppState): def render_timeline_tab(state: AppState):
@@ -226,7 +336,7 @@ def render_timeline_tab(state: AppState):
'text-info q-pa-sm') 'text-info q-pa-sm')
# --- View mode + Selection toggle --- # --- View mode + Selection toggle ---
with ui.row().classes('w-full items-center q-gutter-md'): with ui.row().classes('w-full items-center q-gutter-md q-mb-md'):
ui.label('Version History').classes('text-h6 col') ui.label('Version History').classes('text-h6 col')
view_mode = ui.toggle( view_mode = ui.toggle(
['Horizontal', 'Vertical', 'Linear Log'], ['Horizontal', 'Vertical', 'Linear Log'],
@@ -249,11 +359,10 @@ def render_timeline_tab(state: AppState):
if selection_mode.value and state.timeline_selected_nodes: if selection_mode.value and state.timeline_selected_nodes:
_render_batch_delete(htree, data, file_path, state, render_timeline.refresh) _render_batch_delete(htree, data, file_path, state, render_timeline.refresh)
ui.separator() with ui.card().classes('w-full q-pa-md q-mt-md'):
_render_node_manager(
_render_node_manager( all_nodes, htree, data, file_path,
all_nodes, htree, data, file_path, _restore_and_refresh, render_timeline.refresh)
_restore_and_refresh, render_timeline.refresh)
def _toggle_select(nid, checked): def _toggle_select(nid, checked):
if checked: if checked:
@@ -288,7 +397,7 @@ def _render_graphviz(dot_source: str):
def _restore_node(data, node, htree, file_path, state: AppState): def _restore_node(data, node, htree, file_path, state: AppState):
"""Restore a history node as the current version.""" """Restore a history node as the current version."""
node_data = node['data'] node_data = copy.deepcopy(node['data'])
if KEY_BATCH_DATA not in node_data and KEY_BATCH_DATA in data: if KEY_BATCH_DATA not in node_data and KEY_BATCH_DATA in data:
del data[KEY_BATCH_DATA] del data[KEY_BATCH_DATA]
data.update(node_data) data.update(node_data)
@@ -300,9 +409,8 @@ def _restore_node(data, node, htree, file_path, state: AppState):
ui.notify('Restored!', type='positive') ui.notify('Restored!', type='positive')
def _render_data_preview(selected_node_id, htree): def _render_data_preview(nid, htree):
"""Render a read-only preview of the selected node's data.""" """Render a read-only preview of the selected node's data."""
nid = selected_node_id.value
if not nid or nid not in htree.nodes: if not nid or nid not in htree.nodes:
ui.label('No node selected.').classes('text-caption') ui.label('No node selected.').classes('text-caption')
return return