Update tab_gallery_sorter.py
This commit is contained in:
@@ -3,62 +3,71 @@ import os
|
||||
from engine import SorterEngine
|
||||
|
||||
def render(quality):
|
||||
# 1. Configuration Header
|
||||
st.header("🖼️ Gallery Staging Sorter")
|
||||
# 1. Path Configuration for Tab 5
|
||||
# These paths are saved independently in the DB
|
||||
profiles = SorterEngine.load_profiles()
|
||||
active_profile = st.session_state.get('active_profile', 'Default')
|
||||
p_data = profiles.get(active_profile, {})
|
||||
|
||||
st.subheader("🖼️ Gallery Staging Sorter")
|
||||
c1, c2 = st.columns(2)
|
||||
path_s = c1.text_input("Source Gallery Path", key="t5_src")
|
||||
path_o = c2.text_input("Final Output Root", key="t5_out")
|
||||
path_s = c1.text_input("Source Gallery Folder", value=p_data.get("tab5_source", "/storage"), key="t5_s_in")
|
||||
path_o = c2.text_input("Final Output Root", value=p_data.get("tab5_out", "/storage"), key="t5_o_in")
|
||||
|
||||
recursive = st.checkbox("Include Subfolders", value=True)
|
||||
cleanup = st.radio("Unmarked Files Action:", ["Keep in Source", "Move to Unused", "Delete Permanent"], horizontal=True)
|
||||
# Save independent paths if they change
|
||||
if path_s != p_data.get("tab5_source") or path_o != p_data.get("tab5_out"):
|
||||
SorterEngine.save_tab_paths(active_profile, t5_s=path_s, t5_o=path_o)
|
||||
|
||||
# Sorting Options
|
||||
col_opt1, col_opt2 = st.columns(2)
|
||||
recursive = col_opt1.checkbox("Include Subfolders (Recursive Scan)", value=True)
|
||||
cleanup = col_opt2.radio("Action for Unmarked Files:", ["Keep in Source", "Move to Unused", "Delete Permanent"], horizontal=True)
|
||||
|
||||
if not path_s or not os.path.exists(path_s):
|
||||
st.info("Select a valid source folder to begin.")
|
||||
st.info("Select a valid source folder to view the gallery.")
|
||||
return
|
||||
|
||||
# 2. Sidebar for Categories
|
||||
with st.sidebar:
|
||||
st.divider()
|
||||
st.subheader("📁 Staging Categories")
|
||||
cats = SorterEngine.get_categories()
|
||||
selected_cat = st.radio("Active Category", cats)
|
||||
|
||||
new_cat = st.text_input("Add Category")
|
||||
if st.button("➕ Add"):
|
||||
SorterEngine.add_category(new_cat)
|
||||
st.rerun()
|
||||
# 2. Category Selection (Sidebar logic)
|
||||
cats = SorterEngine.get_categories()
|
||||
selected_cat = st.sidebar.radio("🏷️ Target Category Tag", cats, key="t5_cat_sel")
|
||||
|
||||
# 3. Gallery Display
|
||||
images = SorterEngine.get_images(path_s, recursive=recursive)
|
||||
staged = SorterEngine.get_staged_data()
|
||||
|
||||
st.write(f"Total Images: {len(images)} | Staged: {len([i for i in staged.values() if i['marked']])}")
|
||||
st.write(f"**Images Found:** {len(images)} | **Pending Renames:** {len(staged)}")
|
||||
|
||||
# Display images in a grid
|
||||
# Display Grid (4 items wide)
|
||||
cols = st.columns(4)
|
||||
for idx, img_path in enumerate(images):
|
||||
with cols[idx % 4]:
|
||||
is_staged = img_path in staged
|
||||
border_style = "4px solid green" if is_staged else "1px solid gray"
|
||||
|
||||
# Clickable Image logic
|
||||
st.image(SorterEngine.compress_for_web(img_path, quality),
|
||||
caption=os.path.basename(img_path))
|
||||
# Show visual status (Marked vs Unmarked)
|
||||
caption = f"✅ {staged[img_path]['name']}" if is_staged else os.path.basename(img_path)
|
||||
|
||||
if st.button("Tag" if not is_staged else "Untag", key=f"tag_{idx}"):
|
||||
st.image(SorterEngine.compress_for_web(img_path, quality), caption=caption)
|
||||
|
||||
# Tagging logic: Renames in DB, not yet on disk
|
||||
if st.button("Tag as " + selected_cat if not is_staged else "Remove Tag", key=f"t5_btn_{idx}"):
|
||||
if not is_staged:
|
||||
# Calculate new name based on category count
|
||||
ext = os.path.splitext(img_path)[1]
|
||||
# Logic to count current category items for the suffix
|
||||
cat_count = len([v for v in staged.values() if v['cat'] == selected_cat]) + 1
|
||||
new_name = f"{selected_cat}_{cat_count:03d}{ext}"
|
||||
SorterEngine.stage_image(img_path, selected_cat, new_name)
|
||||
else:
|
||||
# Logic to remove from staging...
|
||||
# To un-tag, we would add a delete_staging method (optional)
|
||||
pass
|
||||
st.rerun()
|
||||
|
||||
st.divider()
|
||||
|
||||
# 4. Commit to Disk
|
||||
if st.button("🚀 APPLY ALL CHANGES TO DISK", type="primary", use_container_width=True):
|
||||
SorterEngine.commit_staging(path_o, cleanup)
|
||||
st.success("Files successfully moved and renamed on disk!")
|
||||
st.rerun()
|
||||
if staged:
|
||||
SorterEngine.commit_staging(path_o, cleanup, source_root=path_s)
|
||||
st.success(f"Successfully processed {len(staged)} images and applied cleanup!")
|
||||
st.rerun()
|
||||
else:
|
||||
st.error("No images have been tagged for processing.")
|
||||
Reference in New Issue
Block a user