Update tab_gallery_sorter.py

This commit is contained in:
2026-01-19 16:32:35 +01:00
parent 13818737e2
commit 40453dad94

View File

@@ -65,13 +65,7 @@ def cb_set_page(page_idx):
st.session_state.t5_page = page_idx st.session_state.t5_page = page_idx
def cb_slider_change(key): def cb_slider_change(key):
"""
Updates the page number from the slider.
Adjusts for 1-based display (Slider=1 -> Page=0).
"""
# Get the value from the widget
val = st.session_state[key] val = st.session_state[key]
# Update the global page index (0-based)
st.session_state.t5_page = val - 1 st.session_state.t5_page = val - 1
@@ -112,6 +106,22 @@ def view_tag_preview(img_path, title):
else: else:
st.error(f"Could not load image: {img_path}") st.error(f"Could not load image: {img_path}")
@st.cache_data(show_spinner=False)
def get_cached_green_dots(all_images, page_size, staged_keys):
"""
Calculates which pages have tags.
Cached based on the specific keys in the staging area.
"""
# We reconstruct the set of staged keys from the frozenset
staged_set = set(staged_keys)
tagged_pages = set()
# Efficient O(N) scan ONLY when tagging changes
for idx, img_path in enumerate(all_images):
if img_path in staged_set:
tagged_pages.add(idx // page_size)
return tagged_pages
# ========================================== # ==========================================
# 3. FRAGMENTS # 3. FRAGMENTS
# ========================================== # ==========================================
@@ -239,67 +249,58 @@ def render_sidebar_content(path_o):
# NOTE: Do NOT use @st.fragment here. # NOTE: Do NOT use @st.fragment here.
# Navigation controls must trigger a full app rerun to load the new batch of images. # Navigation controls must trigger a full app rerun to load the new batch of images.
def render_pagination_carousel(key_suffix, total_pages, all_images, page_size): # CHANGED: Added 'tagged_pages_set' to arguments
def render_pagination_carousel(key_suffix, total_pages, current_page, tagged_pages_set):
""" """
Renders pagination with 1-based indexing and smooth callbacks. Renders pagination. No calculation here—just pure UI rendering.
""" """
# Safety Check if total_pages <= 1: return
if total_pages <= 1:
return
current_page = st.session_state.t5_page # 1. Rapid Seeker Slider (1-BASED)
# 1. Get Tagged Pages (for the Green Dot)
tagged_pages_set = SorterEngine.get_tagged_page_indices(all_images, page_size)
# 2. Rapid Seeker Slider (1-BASED)
# We set min=1 and max=total_pages so it looks human-readable.
# The callback 'cb_slider_change' handles the -1 conversion.
st.slider( st.slider(
"Rapid Navigation", "Rapid Navigation",
min_value=1, min_value=1, max_value=total_pages, value=current_page + 1, step=1,
max_value=total_pages, key=f"slider_{key_suffix}", label_visibility="collapsed",
value=current_page + 1,
step=1,
key=f"slider_{key_suffix}",
label_visibility="collapsed",
on_change=cb_slider_change, args=(f"slider_{key_suffix}",) on_change=cb_slider_change, args=(f"slider_{key_suffix}",)
) )
# 3. Button Window Logic # 2. Window Logic (Calculate range of buttons to show)
window_radius = 2 window_radius = 2
start_p = max(0, current_page - window_radius) start_p = max(0, current_page - window_radius)
end_p = min(total_pages, current_page + window_radius + 1) end_p = min(total_pages, current_page + window_radius + 1)
# Keep the window width constant near edges # Adjust window near edges to keep width constant
if current_page < window_radius: if current_page < window_radius:
end_p = min(total_pages, 5) end_p = min(total_pages, 5)
elif current_page > total_pages - window_radius - 1: elif current_page > total_pages - window_radius - 1:
start_p = max(0, total_pages - 5) start_p = max(0, total_pages - 5)
num_page_buttons = end_p - start_p num_page_buttons = end_p - start_p
if num_page_buttons < 1: return # Safety check if page count is small
if num_page_buttons < 1:
start_p = 0
end_p = total_pages
num_page_buttons = total_pages
# 4. Render Buttons # 3. Render Buttons
# We create columns: [Prev] + [1] [2] [3] ... + [Next]
cols = st.columns([1] + [1] * num_page_buttons + [1]) cols = st.columns([1] + [1] * num_page_buttons + [1])
# PREV # --- PREV BUTTON ---
with cols[0]: with cols[0]:
st.button("", disabled=(current_page == 0), st.button("", disabled=(current_page == 0),
on_click=cb_change_page, args=(-1,), on_click=cb_change_page, args=(-1,),
key=f"prev_{key_suffix}", use_container_width=True) key=f"prev_{key_suffix}", use_container_width=True)
# NUMBERED BUTTONS (1-BASED LABELS) # --- NUMBERED BUTTONS ---
for i, p_idx in enumerate(range(start_p, end_p)): for i, p_idx in enumerate(range(start_p, end_p)):
with cols[i + 1]: with cols[i + 1]:
# Human readable label (Page 0 -> "1")
label = str(p_idx + 1) label = str(p_idx + 1)
# Add Green Dot if this page has tagged items
# Green Dot Indicator
if p_idx in tagged_pages_set: if p_idx in tagged_pages_set:
label += " 🟢" label += " 🟢"
# Highlight Active Page # Highlight Current Page
btn_type = "primary" if p_idx == current_page else "secondary" btn_type = "primary" if p_idx == current_page else "secondary"
st.button(label, type=btn_type, st.button(label, type=btn_type,
@@ -307,7 +308,7 @@ def render_pagination_carousel(key_suffix, total_pages, all_images, page_size):
use_container_width=True, use_container_width=True,
on_click=cb_set_page, args=(p_idx,)) on_click=cb_set_page, args=(p_idx,))
# NEXT # --- NEXT BUTTON ---
with cols[-1]: with cols[-1]:
st.button("", disabled=(current_page >= total_pages - 1), st.button("", disabled=(current_page >= total_pages - 1),
on_click=cb_change_page, args=(1,), on_click=cb_change_page, args=(1,),
@@ -454,7 +455,7 @@ def render(quality, profile_name):
st.info("No images found.") st.info("No images found.")
return return
# Pagination Math # Pagination Math
total_items = len(all_images) total_items = len(all_images)
total_pages = math.ceil(total_items / page_size) total_pages = math.ceil(total_items / page_size)
if st.session_state.t5_page >= total_pages: st.session_state.t5_page = max(0, total_pages - 1) if st.session_state.t5_page >= total_pages: st.session_state.t5_page = max(0, total_pages - 1)
@@ -464,14 +465,26 @@ def render(quality, profile_name):
end_idx = start_idx + page_size end_idx = start_idx + page_size
current_batch = all_images[start_idx:end_idx] current_batch = all_images[start_idx:end_idx]
# --- RENDER UI --- # --- 1. CALCULATE GREEN DOTS ONCE (Optimized) ---
staged_data = SorterEngine.get_staged_data()
# We pass 'frozenset' so the cache works efficiently
green_dots_set = get_cached_green_dots(all_images, page_size, frozenset(staged_data.keys()))
# --- 2. RENDER UI ---
st.divider() st.divider()
render_pagination_carousel("top", total_pages, all_images, page_size)
# CORRECTED CALL: Pass 'current_page' and 'green_dots_set'
render_pagination_carousel("top", total_pages, st.session_state.t5_page, green_dots_set)
render_gallery_grid(current_batch, quality, grid_cols, path_o) render_gallery_grid(current_batch, quality, grid_cols, path_o)
st.divider() st.divider()
render_pagination_carousel("bot", total_pages, all_images, page_size)
# CORRECTED CALL
render_pagination_carousel("bot", total_pages, st.session_state.t5_page, green_dots_set)
st.divider()
render_batch_actions(current_batch, path_o, st.session_state.t5_page + 1, path_s)
st.divider() st.divider()
render_batch_actions(current_batch, path_o, st.session_state.t5_page + 1, path_s) render_batch_actions(current_batch, path_o, st.session_state.t5_page + 1, path_s)