diff --git a/app/frontend/ui.py b/app/frontend/ui.py index 8be766e..b68014c 100644 --- a/app/frontend/ui.py +++ b/app/frontend/ui.py @@ -23,7 +23,7 @@ timeout_setting = os.getenv("MINDNET_API_TIMEOUT") or os.getenv("MINDNET_LLM_TIM API_TIMEOUT = float(timeout_setting) if timeout_setting else 300.0 # --- PAGE SETUP --- -st.set_page_config(page_title="mindnet v2.3.8", page_icon="🧠", layout="wide") +st.set_page_config(page_title="mindnet v2.3.9", page_icon="🧠", layout="wide") # --- CSS STYLING --- st.markdown(""" @@ -225,7 +225,7 @@ def submit_feedback(query_id, node_id, score, comment=None): def render_sidebar(): with st.sidebar: st.title("🧠 mindnet") - st.caption("v2.3.8 | Fixed State Sync") + st.caption("v2.3.9 | Stable ID Fix") mode = st.radio("Modus", ["💬 Chat", "📝 Manueller Editor"], index=0) st.divider() st.subheader("⚙️ Settings") @@ -240,15 +240,23 @@ def render_sidebar(): return mode, top_k, explain def render_draft_editor(msg): - qid = msg.get('query_id', str(uuid.uuid4())) + # --- STABLE ID FIX (Der entscheidende Teil) --- + # Wir prüfen, ob die Nachricht schon eine ID hat. Wenn nicht, erzeugen wir eine + # und SPEICHERN sie zurück in das msg-Objekt (das Teil von session_state ist). + # So bleibt die ID über Reruns hinweg identisch. + if "query_id" not in msg or not msg["query_id"]: + msg["query_id"] = str(uuid.uuid4()) + + qid = msg["query_id"] key_base = f"draft_{qid}" # State Keys data_meta_key = f"{key_base}_data_meta" data_sugg_key = f"{key_base}_data_suggestions" widget_body_key = f"{key_base}_widget_body" + data_body_key = f"{key_base}_data_body" - # --- 1. INIT STATE (Nur einmalig) --- + # --- 1. INIT STATE (Nur einmalig pro stabiler ID) --- if f"{key_base}_init" not in st.session_state: meta, body = parse_markdown_draft(msg["content"]) @@ -258,24 +266,32 @@ def render_draft_editor(msg): tags = meta.get("tags", []) meta["tags_str"] = ", ".join(tags) if isinstance(tags, list) else str(tags) + # Persistent Data st.session_state[data_meta_key] = meta st.session_state[data_sugg_key] = [] + st.session_state[data_body_key] = body.strip() - # WICHTIG: Wir initialisieren den Widget Key direkt! + # Widget Init (wichtig: Hier wird der "Default Value" des Widgets gesetzt) st.session_state[widget_body_key] = body.strip() st.session_state[f"{key_base}_init"] = True - # --- CALLBACKS (Modifizieren direkt den Widget-Key) --- - + # --- CALLBACKS --- + def _sync_body(): + # Sync vom Widget zurück in den persistenten Speicher + st.session_state[data_body_key] = st.session_state[widget_body_key] + def _insert_text(text_to_insert): + # Einfügen in Widget State current = st.session_state[widget_body_key] st.session_state[widget_body_key] = f"{current}\n\n{text_to_insert}" - # Kein Rerun nötig, Button-Klick macht das implizit + # Sync auch data_body + st.session_state[data_body_key] = st.session_state[widget_body_key] def _remove_text(text_to_remove): current = st.session_state[widget_body_key] st.session_state[widget_body_key] = current.replace(text_to_remove, "").strip() + st.session_state[data_body_key] = st.session_state[widget_body_key] # --- UI LAYOUT --- st.markdown(f'