Update tab_gallery_sorter.py
This commit is contained in:
@@ -5,29 +5,23 @@ from engine import SorterEngine
|
|||||||
def render(quality, profile_name):
|
def render(quality, profile_name):
|
||||||
st.subheader("🖼️ Gallery Staging Sorter")
|
st.subheader("🖼️ Gallery Staging Sorter")
|
||||||
|
|
||||||
# 1. Load data for THIS specific tab
|
# 1. Load Workspace Settings
|
||||||
profiles = SorterEngine.load_profiles()
|
profiles = SorterEngine.load_profiles()
|
||||||
p_data = profiles.get(profile_name, {})
|
p_data = profiles.get(profile_name, {})
|
||||||
|
|
||||||
# Path Inputs with explicit string fallbacks
|
|
||||||
c1, c2 = st.columns(2)
|
c1, c2 = st.columns(2)
|
||||||
t5_s = p_data.get("tab5_source") or "/storage"
|
path_s = c1.text_input("📁 Source Gallery Folder", value=p_data.get("tab5_source") or "/storage", key="t5_s_path")
|
||||||
t5_o = p_data.get("tab5_out") or "/storage"
|
path_o = c2.text_input("🎯 Final Output Root", value=p_data.get("tab5_out") or "/storage", key="t5_o_path")
|
||||||
|
|
||||||
path_s = c1.text_input("📁 Source Gallery Folder", value=t5_s, key="t5_s_path")
|
if path_s != p_data.get("tab5_source") or path_o != p_data.get("tab5_out"):
|
||||||
path_o = c2.text_input("🎯 Final Output Root", value=t5_o, key="t5_o_path")
|
if st.button("💾 Save Gallery Paths"):
|
||||||
|
|
||||||
# Save button only appears if paths changed
|
|
||||||
if path_s != t5_s or path_o != t5_o:
|
|
||||||
if st.button("💾 Update Workspace Paths"):
|
|
||||||
SorterEngine.save_tab_paths(profile_name, t5_s=path_s, t5_o=path_o)
|
SorterEngine.save_tab_paths(profile_name, t5_s=path_s, t5_o=path_o)
|
||||||
st.rerun()
|
st.rerun()
|
||||||
|
|
||||||
# --- 2. CATEGORY MANAGEMENT (Sorted Sidebar) ---
|
# --- 2. SIDEBAR CATEGORIES ---
|
||||||
with st.sidebar:
|
with st.sidebar:
|
||||||
st.divider()
|
st.divider()
|
||||||
st.subheader("🏷️ Category Manager")
|
st.subheader("🏷️ Category Manager")
|
||||||
|
|
||||||
new_cat = st.text_input("Quick Add Category", key="t5_add_cat")
|
new_cat = st.text_input("Quick Add Category", key="t5_add_cat")
|
||||||
if st.button("➕ Add", use_container_width=True):
|
if st.button("➕ Add", use_container_width=True):
|
||||||
if new_cat:
|
if new_cat:
|
||||||
@@ -35,59 +29,64 @@ def render(quality, profile_name):
|
|||||||
st.rerun()
|
st.rerun()
|
||||||
|
|
||||||
st.divider()
|
st.divider()
|
||||||
# Categories are now automatically sorted A-Z by the engine
|
|
||||||
cats = SorterEngine.get_categories()
|
cats = SorterEngine.get_categories()
|
||||||
if not cats:
|
if not cats:
|
||||||
st.warning("No categories found.")
|
st.warning("No categories found.")
|
||||||
return
|
return
|
||||||
|
|
||||||
selected_cat = st.radio("Active Tag:", cats, key="t5_active_tag")
|
selected_cat = st.radio("Active Tag:", cats, key="t5_active_tag")
|
||||||
|
|
||||||
# --- 3. GALLERY ---
|
|
||||||
if not os.path.exists(path_s):
|
if not os.path.exists(path_s):
|
||||||
st.info("Please enter a valid Source Path to load images.")
|
st.info("Please enter a valid Source Path.")
|
||||||
return
|
return
|
||||||
|
|
||||||
images = SorterEngine.get_images(path_s, recursive=True)
|
# --- 3. THE FRAGMENTED GALLERY ---
|
||||||
staged = SorterEngine.get_staged_data()
|
# This decorator prevents the whole app from rerunning when tagging
|
||||||
|
@st.fragment
|
||||||
st.write(f"Images Found: **{len(images)}** | Tagged: **{len(staged)}**")
|
def render_gallery():
|
||||||
|
images = SorterEngine.get_images(path_s, recursive=True)
|
||||||
|
staged = SorterEngine.get_staged_data()
|
||||||
|
|
||||||
|
st.write(f"Images: **{len(images)}** | Tagged: **{len(staged)}**")
|
||||||
|
|
||||||
|
cols = st.columns(4)
|
||||||
|
for idx, img_path in enumerate(images):
|
||||||
|
with cols[idx % 4]:
|
||||||
|
is_staged = img_path in staged
|
||||||
|
|
||||||
|
# Visual Tag Indicators
|
||||||
|
if is_staged:
|
||||||
|
info = staged[img_path]
|
||||||
|
st.markdown(f"**✅ {info['cat']}**")
|
||||||
|
label = info['name']
|
||||||
|
else:
|
||||||
|
label = os.path.basename(img_path)
|
||||||
|
|
||||||
|
st.image(SorterEngine.compress_for_web(img_path, quality), caption=label)
|
||||||
|
|
||||||
|
# Action Buttons inside Fragment
|
||||||
|
if not is_staged:
|
||||||
|
if st.button(f"Tag: {selected_cat}", key=f"tag_{idx}"):
|
||||||
|
ext = os.path.splitext(img_path)[1]
|
||||||
|
count = len([v for v in staged.values() if v['cat'] == selected_cat]) + 1
|
||||||
|
new_name = f"{selected_cat}_{count:03d}{ext}"
|
||||||
|
SorterEngine.stage_image(img_path, selected_cat, new_name)
|
||||||
|
st.rerun(scope="fragment") # Reruns ONLY the gallery
|
||||||
|
else:
|
||||||
|
if st.button("❌ Remove", key=f"clear_{idx}"):
|
||||||
|
SorterEngine.clear_staged_item(img_path)
|
||||||
|
st.rerun(scope="fragment")
|
||||||
|
|
||||||
cols = st.columns(4)
|
# Call the fragmented gallery
|
||||||
for idx, img_path in enumerate(images):
|
render_gallery()
|
||||||
with cols[idx % 4]:
|
|
||||||
is_staged = img_path in staged
|
|
||||||
|
|
||||||
# Labeling logic
|
|
||||||
if is_staged:
|
|
||||||
info = staged[img_path]
|
|
||||||
st.success(f"TAGGED: {info['cat']}")
|
|
||||||
label = f"Renaming to: {info['name']}"
|
|
||||||
else:
|
|
||||||
label = os.path.basename(img_path)
|
|
||||||
|
|
||||||
st.image(SorterEngine.compress_for_web(img_path, quality), caption=label)
|
|
||||||
|
|
||||||
# Action logic
|
|
||||||
if not is_staged:
|
|
||||||
if st.button(f"Tag: {selected_cat}", key=f"tag_{idx}"):
|
|
||||||
ext = os.path.splitext(img_path)[1]
|
|
||||||
# Logic to ensure numbering starts at 001 for each category session
|
|
||||||
count = len([v for v in staged.values() if v['cat'] == selected_cat]) + 1
|
|
||||||
new_name = f"{selected_cat}_{count:03d}{ext}"
|
|
||||||
SorterEngine.stage_image(img_path, selected_cat, new_name)
|
|
||||||
st.rerun()
|
|
||||||
else:
|
|
||||||
if st.button("❌ Remove Tag", key=f"clear_{idx}"):
|
|
||||||
SorterEngine.clear_staged_item(img_path)
|
|
||||||
st.rerun()
|
|
||||||
|
|
||||||
st.divider()
|
st.divider()
|
||||||
|
|
||||||
# --- 4. APPLY ---
|
# --- 4. APPLY (Outside fragment to ensure full refresh after disk move) ---
|
||||||
cleanup = st.radio("Unmarked Files Action:", ["Keep", "Move to Unused", "Delete"], horizontal=True)
|
cleanup = st.radio("Unmarked Files Action:", ["Keep", "Move to Unused", "Delete"], horizontal=True)
|
||||||
if st.button("🚀 APPLY ALL CHANGES TO DISK", type="primary", use_container_width=True):
|
if st.button("🚀 APPLY ALL CHANGES TO DISK", type="primary", use_container_width=True):
|
||||||
if staged:
|
if staged := SorterEngine.get_staged_data():
|
||||||
SorterEngine.commit_staging(path_o, cleanup, source_root=path_s)
|
SorterEngine.commit_staging(path_o, cleanup, source_root=path_s)
|
||||||
st.success("Successfully processed images!")
|
st.success("Successfully processed images!")
|
||||||
st.rerun()
|
st.rerun()
|
||||||
|
|
||||||
|
render_gallery = render # Mapping for the tab loader
|
||||||
Reference in New Issue
Block a user