This commit is contained in:
team3
2026-06-14 14:55:44 +02:00
parent 2b89e21cd3
commit 143e6d6f7c
7 changed files with 108 additions and 41 deletions

View File

@@ -65,6 +65,7 @@ CREATE TABLE IF NOT EXISTS baustein_progress (
baustein TEXT NOT NULL, baustein TEXT NOT NULL,
gute_antworten INTEGER NOT NULL DEFAULT 0, gute_antworten INTEGER NOT NULL DEFAULT 0,
absolviert TEXT, absolviert TEXT,
verstanden TEXT,
updated_at TEXT NOT NULL, updated_at TEXT NOT NULL,
PRIMARY KEY (topic, baustein) PRIMARY KEY (topic, baustein)
) )
@@ -96,6 +97,10 @@ async def init_db():
await db.execute("ALTER TABLE guides ADD COLUMN step INTEGER") await db.execute("ALTER TABLE guides ADD COLUMN step INTEGER")
except aiosqlite.OperationalError: except aiosqlite.OperationalError:
pass pass
try: # Migration für Bestands-DBs ohne verstanden-Spalte (Mastery-Stufe)
await db.execute("ALTER TABLE baustein_progress ADD COLUMN verstanden TEXT")
except aiosqlite.OperationalError:
pass
# Migration: alte vertiefungen-Tabelle → baustein_texte (Bestand = lange Form, art 'deepdive') # Migration: alte vertiefungen-Tabelle → baustein_texte (Bestand = lange Form, art 'deepdive')
cursor = await db.execute("SELECT name FROM sqlite_master WHERE type = 'table' AND name = 'vertiefungen'") cursor = await db.execute("SELECT name FROM sqlite_master WHERE type = 'table' AND name = 'vertiefungen'")
if await cursor.fetchone(): if await cursor.fetchone():
@@ -333,29 +338,41 @@ async def list_vertiefungen(topic: str) -> dict[str, set[str]]:
async def list_baustein_progress(topic: str) -> list[dict]: async def list_baustein_progress(topic: str) -> list[dict]:
db = await get_db() db = await get_db()
cursor = await db.execute( cursor = await db.execute(
"SELECT baustein, gute_antworten, absolviert FROM baustein_progress WHERE topic = ?", (topic,) "SELECT baustein, gute_antworten, absolviert, verstanden FROM baustein_progress WHERE topic = ?", (topic,)
) )
rows = await cursor.fetchall() rows = await cursor.fetchall()
return [{"baustein": b, "gute_antworten": n, "absolviert": a} for b, n, a in rows] return [{"baustein": b, "gute_antworten": n, "absolviert": a, "verstanden": v} for b, n, a, v in rows]
async def add_gute_antwort(topic: str, baustein: str) -> int: async def set_baustein_score(topic: str, baustein: str, score: int) -> int:
"""Zählt eine gut bewertete Antwort und liefert den neuen Stand.""" """Setzt den Score absolut (vom Aufrufer geclampt) und liefert ihn zurück."""
db = await get_db() db = await get_db()
await db.execute( await db.execute(
"""INSERT INTO baustein_progress (topic, baustein, gute_antworten, updated_at) """INSERT INTO baustein_progress (topic, baustein, gute_antworten, updated_at)
VALUES (?, ?, 1, ?) VALUES (?, ?, ?, ?)
ON CONFLICT(topic, baustein) DO UPDATE SET ON CONFLICT(topic, baustein) DO UPDATE SET
gute_antworten = gute_antworten + 1, updated_at = excluded.updated_at""", gute_antworten = excluded.gute_antworten, updated_at = excluded.updated_at""",
(topic, baustein, _now()), (topic, baustein, score, _now()),
) )
await db.commit() await db.commit()
cursor = await db.execute( return score
"SELECT gute_antworten FROM baustein_progress WHERE topic = ? AND baustein = ?",
(topic, baustein),
async def set_baustein_verstanden(topic: str, baustein: str) -> bool:
"""Markiert verstanden (Mastery); True nur beim ersten Mal. Sticky wie absolviert."""
db = await get_db()
now = _now()
await db.execute(
"INSERT OR IGNORE INTO baustein_progress (topic, baustein, gute_antworten, updated_at) VALUES (?, ?, 0, ?)",
(topic, baustein, now),
) )
row = await cursor.fetchone() cursor = await db.execute(
return row[0] if row else 0 "UPDATE baustein_progress SET verstanden = ?, updated_at = ? "
"WHERE topic = ? AND baustein = ? AND verstanden IS NULL",
(now, now, topic, baustein),
)
await db.commit()
return cursor.rowcount > 0
async def set_baustein_absolviert(topic: str, baustein: str) -> bool: async def set_baustein_absolviert(topic: str, baustein: str) -> bool:

View File

@@ -19,13 +19,29 @@ from textkit import _norm_titel
log = logging.getLogger("creator.lernen") log = logging.getLogger("creator.lernen")
NOETIG = 3 # gute Antworten bis "absolviert" NOETIG = 3 # gute Antworten bis "absolviert" (Tier 1)
MASTERY = 10 # Score bis "verstanden" (Tier 2)
VERTIEFUNG_TIMEOUT = 600 VERTIEFUNG_TIMEOUT = 600
CHAT_TIMEOUT = 240 CHAT_TIMEOUT = 240
PRUEFUNG_TIMEOUT = 120 # kurze JSON-Turns; deckelt die Serien-Latenz pro Prüfungs-Schritt PRUEFUNG_TIMEOUT = 120 # kurze JSON-Turns; deckelt die Serien-Latenz pro Prüfungs-Schritt
KRITIK_MAX_RUNDEN = 2 # Generator → Kritiker → ggf. Neu, höchstens so oft KRITIK_MAX_RUNDEN = 2 # Generator → Kritiker → ggf. Neu, höchstens so oft
def score_berechnen(score_vor_frage: int, gut: bool, tier2: bool, absolviert: bool) -> int:
"""Neuer Score nach einer Antwort · driftfrei (immer aus dem Basis-Score gerechnet).
Tier 1 (tier2=False): +1 bei richtig, KEINE Strafe bei falsch, Deckel NOETIG (3).
Tier 2 (tier2=True): +1 bei richtig (Deckel MASTERY=10), 1 bei falsch.
Boden ist NOETIG, sobald der Baustein absolviert ist — sonst 0 (kann nicht
unter 3 fallen, absolviert bleibt erhalten). Re-Bewertung nutzt denselben
Basis-Score und ersetzt so das vorige Ergebnis (kein Doppelzählen).
"""
delta = 1 if gut else (-1 if tier2 else 0)
floor = NOETIG if absolviert else 0
cap = MASTERY if tier2 else NOETIG
return max(floor, min(cap, score_vor_frage + delta))
def _transcript(messages: list[dict]) -> str: def _transcript(messages: list[dict]) -> str:
return "\n".join( return "\n".join(
f"{'Nutzer' if m.get('role') == 'user' else 'Assistent'}: {m.get('content', '')}" f"{'Nutzer' if m.get('role') == 'user' else 'Assistent'}: {m.get('content', '')}"

View File

@@ -194,7 +194,8 @@ class BausteinPruefungRequest(BaseModel):
aktion: Literal["frage", "diskussion", "antwort"] = "frage" aktion: Literal["frage", "diskussion", "antwort"] = "frage"
frage: str = Field(default="", max_length=2000) # aktuell geprüfte Frage (für diskussion/antwort) frage: str = Field(default="", max_length=2000) # aktuell geprüfte Frage (für diskussion/antwort)
letzte_bewertung: str = Field(default="", max_length=2000) # Feedback der letzten Bewertung (Kontext für diskussion) letzte_bewertung: str = Field(default="", max_length=2000) # Feedback der letzten Bewertung (Kontext für diskussion)
frage_schon_gut: bool = False # diese Frage wurde schon einmal "gut" bewertet → nicht doppelt zählen score_vor_frage: int = 0 # Score, als die Frage gestellt wurde → driftfreies (Re-)Bewerten
tier2: bool = False # Mastery-Modus (ganzer Guide absolviert) → 1 bei falsch, Deckel 10
messages: list[ChatMessage] = [] # Dialog bisher; leer = erste Frage messages: list[ChatMessage] = [] # Dialog bisher; leer = erste Frage
provider: ProviderType = "claude" provider: ProviderType = "claude"
@@ -206,11 +207,13 @@ class BausteinPruefungResponse(BaseModel):
bewertung: Literal["gut", "schlecht"] | None = None bewertung: Literal["gut", "schlecht"] | None = None
gute_antworten: int gute_antworten: int
absolviert: bool absolviert: bool
verstanden: bool = False
class BausteinLernstand(BaseModel): class BausteinLernstand(BaseModel):
gute_antworten: int gute_antworten: int
absolviert: bool absolviert: bool
verstanden: bool
vertiefung: bool vertiefung: bool
deepdive: bool deepdive: bool

View File

@@ -14,11 +14,11 @@ from database import (
list_progress, set_progress, delete_progress, list_progress, set_progress, delete_progress,
create_element, list_elements, get_element, update_element, delete_element, create_element, list_elements, get_element, update_element, delete_element,
get_vertiefung, set_vertiefung, list_vertiefungen, get_vertiefung, set_vertiefung, list_vertiefungen,
list_baustein_progress, add_gute_antwort, set_baustein_absolviert, delete_baustein_daten, list_baustein_progress, set_baustein_score, set_baustein_absolviert, set_baustein_verstanden, delete_baustein_daten,
) )
from bausteine import generate_bausteine, cancel_bausteine, bausteine_status, active_bausteine, reset_bausteine from bausteine import generate_bausteine, cancel_bausteine, bausteine_status, active_bausteine, reset_bausteine
from elements import generate_element, chat_with_guide, chat_with_element, check_element, style_element, refine_suggestion from elements import generate_element, chat_with_guide, chat_with_element, check_element, style_element, refine_suggestion
from lernen import NOETIG, baustein_chat, baustein_diskussion, baustein_element_anlegen, pruefung_bewertung, pruefung_frage, vertiefung_generieren from lernen import NOETIG, MASTERY, baustein_chat, baustein_diskussion, baustein_element_anlegen, pruefung_bewertung, pruefung_frage, score_berechnen, vertiefung_generieren
from guide import generate_guide, guide_slot_dateien from guide import generate_guide, guide_slot_dateien
from pipeline import cancel_guide from pipeline import cancel_guide
from regeln import FORMATE, formate_stats, guide_lock, ist_absolviert, lade_lernstand from regeln import FORMATE, formate_stats, guide_lock, ist_absolviert, lade_lernstand
@@ -161,6 +161,7 @@ async def baustein_lernstand(topic: str):
p["baustein"]: { p["baustein"]: {
"gute_antworten": p["gute_antworten"], "gute_antworten": p["gute_antworten"],
"absolviert": p["absolviert"] is not None, "absolviert": p["absolviert"] is not None,
"verstanden": p["verstanden"] is not None,
"vertiefung": "vertiefung" in texte.get(p["baustein"], set()), "vertiefung": "vertiefung" in texte.get(p["baustein"], set()),
"deepdive": "deepdive" in texte.get(p["baustein"], set()), "deepdive": "deepdive" in texte.get(p["baustein"], set()),
} }
@@ -169,7 +170,7 @@ async def baustein_lernstand(topic: str):
for b, arten in texte.items(): for b, arten in texte.items():
if b not in bausteine: if b not in bausteine:
bausteine[b] = { bausteine[b] = {
"gute_antworten": 0, "absolviert": False, "gute_antworten": 0, "absolviert": False, "verstanden": False,
"vertiefung": "vertiefung" in arten, "deepdive": "deepdive" in arten, "vertiefung": "vertiefung" in arten, "deepdive": "deepdive" in arten,
} }
return {"bausteine": bausteine} return {"bausteine": bausteine}
@@ -208,10 +209,11 @@ async def baustein_chat_route(req: BausteinChatRequest):
async def baustein_pruefung_route(req: BausteinPruefungRequest): async def baustein_pruefung_route(req: BausteinPruefungRequest):
stand = next( stand = next(
(p for p in await list_baustein_progress(req.topic) if p["baustein"] == req.baustein), (p for p in await list_baustein_progress(req.topic) if p["baustein"] == req.baustein),
{"gute_antworten": 0, "absolviert": None}, {"gute_antworten": 0, "absolviert": None, "verstanden": None},
) )
gute = stand["gute_antworten"] gute = stand["gute_antworten"]
absolviert = stand["absolviert"] is not None absolviert = stand["absolviert"] is not None
verstanden = stand["verstanden"] is not None
vertiefung = await _bester_text(req.topic, req.baustein) vertiefung = await _bester_text(req.topic, req.baustein)
msgs = [m.model_dump() for m in req.messages] msgs = [m.model_dump() for m in req.messages]
@@ -219,7 +221,7 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
frage = await pruefung_frage(req.topic, req.baustein, req.section, vertiefung, msgs, provider=req.provider) frage = await pruefung_frage(req.topic, req.baustein, req.section, vertiefung, msgs, provider=req.provider)
if frage is None: if frage is None:
raise HTTPException(502, "Frage fehlgeschlagen — bitte erneut versuchen") raise HTTPException(502, "Frage fehlgeschlagen — bitte erneut versuchen")
return {"frage": frage, "gute_antworten": gute, "absolviert": absolviert} return {"frage": frage, "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden}
if req.aktion == "diskussion": if req.aktion == "diskussion":
if not req.frage.strip(): if not req.frage.strip():
@@ -230,7 +232,7 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
) )
if reply is None: if reply is None:
raise HTTPException(502, "Diskussion fehlgeschlagen — bitte erneut versuchen") raise HTTPException(502, "Diskussion fehlgeschlagen — bitte erneut versuchen")
return {"reply": reply, "gute_antworten": gute, "absolviert": absolviert} return {"reply": reply, "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden}
# aktion == "antwort" — mindestens eine Nutzer-Antwort muss im Dialog stehen # aktion == "antwort" — mindestens eine Nutzer-Antwort muss im Dialog stehen
# (nach einer Diskussion endet der Dialog mit dem Tutor; Re-Bewertung bleibt erlaubt). # (nach einer Diskussion endet der Dialog mit dem Tutor; Re-Bewertung bleibt erlaubt).
@@ -244,14 +246,17 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
if data is None: if data is None:
raise HTTPException(502, "Bewertung fehlgeschlagen — bitte erneut versuchen") raise HTTPException(502, "Bewertung fehlgeschlagen — bitte erneut versuchen")
if data["bewertung"] == "gut" and not req.frage_schon_gut: # Score driftfrei aus dem Basis-Score rechnen (Re-Bewertung ersetzt das vorige Ergebnis).
gute = await add_gute_antwort(req.topic, req.baustein) score = score_berechnen(req.score_vor_frage, data["bewertung"] == "gut", req.tier2, absolviert)
if gute >= NOETIG or data["bestanden"]: gute = await set_baustein_score(req.topic, req.baustein, score)
frisch = await set_baustein_absolviert(req.topic, req.baustein) if score >= NOETIG and not absolviert:
absolviert = True absolviert = True
if frisch: if await set_baustein_absolviert(req.topic, req.baustein):
asyncio.create_task(baustein_element_anlegen(req.topic, req.baustein, req.section, req.provider)) asyncio.create_task(baustein_element_anlegen(req.topic, req.baustein, req.section, req.provider))
return {"feedback": data["feedback"], "bewertung": data["bewertung"], "gute_antworten": gute, "absolviert": absolviert} if score >= MASTERY and not verstanden:
await set_baustein_verstanden(req.topic, req.baustein)
verstanden = True
return {"feedback": data["feedback"], "bewertung": data["bewertung"], "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden}
# --- Guides --- # --- Guides ---

View File

@@ -93,12 +93,12 @@ export async function chatBaustein({ topic, baustein, section, messages, provide
export async function pruefeBaustein({ export async function pruefeBaustein({
topic, baustein, section, provider, topic, baustein, section, provider,
aktion = 'frage', frage = '', letzte_bewertung = '', frage_schon_gut = false, messages = [], aktion = 'frage', frage = '', letzte_bewertung = '', score_vor_frage = 0, tier2 = false, messages = [],
}) { }) {
const res = await fetch(`${BASE}/bausteine/pruefung`, { const res = await fetch(`${BASE}/bausteine/pruefung`, {
method: 'POST', method: 'POST',
headers: { 'Content-Type': 'application/json' }, headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ topic, baustein, section, aktion, frage, letzte_bewertung, frage_schon_gut, messages, provider }), body: JSON.stringify({ topic, baustein, section, aktion, frage, letzte_bewertung, score_vor_frage, tier2, messages, provider }),
}) })
return jsonOrThrow(res) return jsonOrThrow(res)
} }

View File

@@ -9,13 +9,15 @@ const props = defineProps({
baustein: { type: String, required: true }, baustein: { type: String, required: true },
section: { type: String, default: '' }, section: { type: String, default: '' },
provider: { type: String, default: 'claude' }, provider: { type: String, default: 'claude' },
status: { type: Object, default: null }, // {gute_antworten, absolviert, vertiefung} status: { type: Object, default: null }, // {gute_antworten, absolviert, verstanden, vertiefung}
tier2: { type: Boolean, default: false }, // Mastery frei (ganzer Guide absolviert)
}) })
const emit = defineEmits(['statusChanged']) const emit = defineEmits(['statusChanged'])
const NOETIG = 3 const NOETIG = 3 // absolviert
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, vertiefung: false, deepdive: false }) const MAX = 10 // verstanden
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, verstanden: false, vertiefung: false, deepdive: false })
// --- Toggle-Bereich --- // --- Toggle-Bereich ---
const activeTab = ref(null) // null | 'vertiefung' | 'deepdive' | 'chat' | 'pruefung' const activeTab = ref(null) // null | 'vertiefung' | 'deepdive' | 'chat' | 'pruefung'
@@ -75,13 +77,13 @@ const pruefPhase = ref('idle')
const pruefLoading = ref(false) const pruefLoading = ref(false)
const aktuelleFrage = ref('') // ankert Bewertung/Diskussion const aktuelleFrage = ref('') // ankert Bewertung/Diskussion
const letztesFeedback = ref('') // Kontext für die Diskussion über eine Bewertung const letztesFeedback = ref('') // Kontext für die Diskussion über eine Bewertung
const frageSchonGut = ref(false) // diese Frage schon "gut" → nicht doppelt zählen const scoreVorFrage = ref(0) // Score, als die aktuelle Frage gestellt wurde → driftfreies (Re-)Bewerten
const pruefMessagesEl = ref(null) const pruefMessagesEl = ref(null)
const pruefInputEl = ref(null) const pruefInputEl = ref(null)
let pruefRun = 0 let pruefRun = 0
function applyPruefung(res) { function applyPruefung(res) {
emit('statusChanged', { ...st.value, gute_antworten: res.gute_antworten, absolviert: res.absolviert }) emit('statusChanged', { ...st.value, gute_antworten: res.gute_antworten, absolviert: res.absolviert, verstanden: res.verstanden })
} }
async function pruefScroll() { async function pruefScroll() {
@@ -122,7 +124,7 @@ function frageAnfordern() {
pruefSenden({ aktion: 'frage' }, (res) => { pruefSenden({ aktion: 'frage' }, (res) => {
aktuelleFrage.value = res.frage aktuelleFrage.value = res.frage
letztesFeedback.value = '' letztesFeedback.value = ''
frageSchonGut.value = false scoreVorFrage.value = res.gute_antworten // Basis für (Re-)Bewertung dieser Frage
pruefMessages.value.push({ role: 'assistant', kind: 'frage', content: res.frage }) pruefMessages.value.push({ role: 'assistant', kind: 'frage', content: res.frage })
pruefPhase.value = 'frage_offen' pruefPhase.value = 'frage_offen'
}) })
@@ -141,7 +143,6 @@ function nachfragen() {
function bewerten(res) { function bewerten(res) {
letztesFeedback.value = res.feedback || '' letztesFeedback.value = res.feedback || ''
if (res.bewertung === 'gut') frageSchonGut.value = true
pruefMessages.value.push({ role: 'assistant', kind: 'feedback', content: res.feedback || '', bewertung: res.bewertung }) pruefMessages.value.push({ role: 'assistant', kind: 'feedback', content: res.feedback || '', bewertung: res.bewertung })
pruefPhase.value = 'bewertet' pruefPhase.value = 'bewertet'
} }
@@ -151,12 +152,12 @@ function antwortAbgeben() {
if (!text || pruefLoading.value) return if (!text || pruefLoading.value) return
pruefMessages.value.push({ role: 'user', kind: 'antwort', content: text }) pruefMessages.value.push({ role: 'user', kind: 'antwort', content: text })
pruefInput.value = '' pruefInput.value = ''
pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, frage_schon_gut: frageSchonGut.value }, bewerten) pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, score_vor_frage: scoreVorFrage.value, tier2: props.tier2 }, bewerten)
} }
function neuBewerten() { function neuBewerten() {
if (pruefLoading.value) return if (pruefLoading.value) return
pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, frage_schon_gut: frageSchonGut.value }, bewerten) pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, score_vor_frage: scoreVorFrage.value, tier2: props.tier2 }, bewerten)
} }
</script> </script>
@@ -174,7 +175,9 @@ function neuBewerten() {
</button> </button>
<button :class="{ active: activeTab === 'pruefung' }" @click="toggle('pruefung')"> <button :class="{ active: activeTab === 'pruefung' }" @click="toggle('pruefung')">
Prüfung Prüfung
<span v-if="st.absolviert" class="bp-chip done"></span> <span v-if="st.verstanden" class="bp-chip gold" title="Verstanden (10/10)"></span>
<span v-else-if="st.absolviert && tier2" class="bp-chip done"> {{ st.gute_antworten }}/{{ MAX }}</span>
<span v-else-if="st.absolviert" class="bp-chip done"></span>
<span v-else-if="st.gute_antworten" class="bp-chip">{{ Math.min(st.gute_antworten, NOETIG) }}/{{ NOETIG }}</span> <span v-else-if="st.gute_antworten" class="bp-chip">{{ Math.min(st.gute_antworten, NOETIG) }}/{{ NOETIG }}</span>
</button> </button>
</div> </div>
@@ -223,7 +226,9 @@ function neuBewerten() {
<!-- Prüfung: gesteuerter Dialog --> <!-- Prüfung: gesteuerter Dialog -->
<div v-else> <div v-else>
<p class="bp-hint"> <p class="bp-hint">
<template v-if="st.absolviert"> Absolviert du kannst dich weiter prüfen lassen.</template> <template v-if="st.verstanden"> Verstanden ({{ st.gute_antworten }}/{{ MAX }}) Gold bleibt dir.</template>
<template v-else-if="st.absolviert && tier2">Mastery: {{ st.gute_antworten }}/{{ MAX }}. Richtig = +1, falsch = 1 (nicht unter {{ NOETIG }}). Bei {{ MAX }} verstanden.</template>
<template v-else-if="st.absolviert"> Absolviert. Mehr ({{ NOETIG }}{{ MAX }}) gibt's, sobald der ganze Guide absolviert ist.</template>
<template v-else>{{ Math.min(st.gute_antworten, NOETIG) }}/{{ NOETIG }} guten Antworten. Frag nach, wenn etwas unklar ist diskutieren ist erlaubt.</template> <template v-else>{{ Math.min(st.gute_antworten, NOETIG) }}/{{ NOETIG }} guten Antworten. Frag nach, wenn etwas unklar ist diskutieren ist erlaubt.</template>
</p> </p>
@@ -295,6 +300,7 @@ function neuBewerten() {
color: var(--text-muted); color: var(--text-muted);
} }
.bp-chip.done { background: var(--success-soft); border-color: var(--success-border); color: var(--success); } .bp-chip.done { background: var(--success-soft); border-color: var(--success-border); color: var(--success); }
.bp-chip.gold { background: color-mix(in srgb, #d4af37 20%, var(--panel)); border-color: #d4af37; color: #8a6d12; }
.bp-panel { .bp-panel {
margin-top: 0.6rem; margin-top: 0.6rem;

View File

@@ -55,6 +55,12 @@ function onBausteinStatus(baustein, status) {
if (status.absolviert && !warAbsolviert) emit('progressChanged') // Locks/Stats neu laden if (status.absolviert && !warAbsolviert) emit('progressChanged') // Locks/Stats neu laden
} }
// Tier 2 (Mastery, Score 3→10) ist frei, sobald ALLE Bausteine des Guides absolviert sind.
const guideAbsolviert = computed(() => {
const secs = (content.value?.chapters || []).flatMap((ch) => ch.sections)
return secs.length > 0 && secs.every((s) => lernstand.value[s.title]?.absolviert)
})
// --- Chat (Mechanik in useChat; Kontext-Extraktion bleibt hier) --- // --- Chat (Mechanik in useChat; Kontext-Extraktion bleibt hier) ---
const chat = useChat((msgs) => { const chat = useChat((msgs) => {
const { section, outline } = extractContext() const { section, outline } = extractContext()
@@ -153,12 +159,13 @@ function extractContext() {
<article <article
v-for="s in ch.sections" v-for="s in ch.sections"
:key="s.num" :key="s.num"
:class="['section-card', isOnePager && s.key ? 'op-card op-' + s.key : '', lernstand[s.title]?.absolviert ? 'absolviert' : '']" :class="['section-card', isOnePager && s.key ? 'op-card op-' + s.key : '', lernstand[s.title]?.verstanden ? 'verstanden' : (lernstand[s.title]?.absolviert ? 'absolviert' : '')]"
:style="isOnePager && s.key ? { gridArea: s.key } : null" :style="isOnePager && s.key ? { gridArea: s.key } : null"
> >
<h3> <h3>
{{ s.title }} {{ s.title }}
<span v-if="lernstand[s.title]?.absolviert" class="baustein-done" title="Prüfung bestanden"> Absolviert</span> <span v-if="lernstand[s.title]?.verstanden" class="baustein-done verstanden" title="Vollständig verstanden (10/10)"> Verstanden</span>
<span v-else-if="lernstand[s.title]?.absolviert" class="baustein-done" title="Prüfung bestanden"> Absolviert</span>
</h3> </h3>
<div class="section-body markdown" v-html="renderMarkdown(s.md)"></div> <div class="section-body markdown" v-html="renderMarkdown(s.md)"></div>
<BausteinPanel <BausteinPanel
@@ -168,6 +175,7 @@ function extractContext() {
:section="s.md" :section="s.md"
:provider="provider" :provider="provider"
:status="lernstand[s.title]" :status="lernstand[s.title]"
:tier2="guideAbsolviert"
@status-changed="(st) => onBausteinStatus(s.title, st)" @status-changed="(st) => onBausteinStatus(s.title, st)"
/> />
</article> </article>
@@ -325,6 +333,18 @@ function extractContext() {
background: color-mix(in srgb, var(--success) 5%, var(--panel)); background: color-mix(in srgb, var(--success) 5%, var(--panel));
} }
/* Verstandene Bausteine (Mastery 10/10): Gold */
.baustein-done.verstanden {
background: color-mix(in srgb, #d4af37 18%, var(--panel));
border-color: #d4af37;
color: #8a6d12;
}
.guide-content:not(.onepager) .section-card.verstanden {
border-color: #d4af37;
border-top: 3px solid #d4af37;
background: color-mix(in srgb, #d4af37 7%, var(--panel));
}
/* Guides: Karten tragen die Kapitel-Akzentfarbe (OnePager hat eigene op-card-Farben) */ /* Guides: Karten tragen die Kapitel-Akzentfarbe (OnePager hat eigene op-card-Farben) */
.guide-content:not(.onepager) .section-card { .guide-content:not(.onepager) .section-card {
border-top: 3px solid color-mix(in srgb, var(--ch-accent, var(--accent)) 65%, transparent); border-top: 3px solid color-mix(in srgb, var(--ch-accent, var(--accent)) 65%, transparent);