update
This commit is contained in:
@@ -66,6 +66,7 @@ CREATE TABLE IF NOT EXISTS baustein_progress (
|
|||||||
gute_antworten INTEGER NOT NULL DEFAULT 0,
|
gute_antworten INTEGER NOT NULL DEFAULT 0,
|
||||||
absolviert TEXT,
|
absolviert TEXT,
|
||||||
verstanden TEXT,
|
verstanden TEXT,
|
||||||
|
gemeistert TEXT,
|
||||||
updated_at TEXT NOT NULL,
|
updated_at TEXT NOT NULL,
|
||||||
PRIMARY KEY (topic, baustein)
|
PRIMARY KEY (topic, baustein)
|
||||||
)
|
)
|
||||||
@@ -101,6 +102,10 @@ async def init_db():
|
|||||||
await db.execute("ALTER TABLE baustein_progress ADD COLUMN verstanden TEXT")
|
await db.execute("ALTER TABLE baustein_progress ADD COLUMN verstanden TEXT")
|
||||||
except aiosqlite.OperationalError:
|
except aiosqlite.OperationalError:
|
||||||
pass
|
pass
|
||||||
|
try: # Migration für Bestands-DBs ohne gemeistert-Spalte (Meisterpfad 25)
|
||||||
|
await db.execute("ALTER TABLE baustein_progress ADD COLUMN gemeistert 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():
|
||||||
@@ -338,10 +343,10 @@ 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, verstanden FROM baustein_progress WHERE topic = ?", (topic,)
|
"SELECT baustein, gute_antworten, absolviert, verstanden, gemeistert FROM baustein_progress WHERE topic = ?", (topic,)
|
||||||
)
|
)
|
||||||
rows = await cursor.fetchall()
|
rows = await cursor.fetchall()
|
||||||
return [{"baustein": b, "gute_antworten": n, "absolviert": a, "verstanden": v} for b, n, a, v in rows]
|
return [{"baustein": b, "gute_antworten": n, "absolviert": a, "verstanden": v, "gemeistert": m} for b, n, a, v, m in rows]
|
||||||
|
|
||||||
|
|
||||||
async def set_baustein_score(topic: str, baustein: str, score: int) -> int:
|
async def set_baustein_score(topic: str, baustein: str, score: int) -> int:
|
||||||
@@ -375,6 +380,23 @@ async def set_baustein_verstanden(topic: str, baustein: str) -> bool:
|
|||||||
return cursor.rowcount > 0
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
|
||||||
|
async def set_baustein_gemeistert(topic: str, baustein: str) -> bool:
|
||||||
|
"""Markiert gemeistert (Meisterpfad, Score 25); True nur beim ersten Mal. Sticky."""
|
||||||
|
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),
|
||||||
|
)
|
||||||
|
cursor = await db.execute(
|
||||||
|
"UPDATE baustein_progress SET gemeistert = ?, updated_at = ? "
|
||||||
|
"WHERE topic = ? AND baustein = ? AND gemeistert 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:
|
||||||
"""Markiert absolviert; True nur beim ersten Mal (steuert den Element-Task)."""
|
"""Markiert absolviert; True nur beim ersten Mal (steuert den Element-Task)."""
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
|
|||||||
@@ -19,26 +19,36 @@ from textkit import _norm_titel
|
|||||||
|
|
||||||
log = logging.getLogger("creator.lernen")
|
log = logging.getLogger("creator.lernen")
|
||||||
|
|
||||||
NOETIG = 3 # gute Antworten bis "absolviert" (Tier 1)
|
NOETIG = 3 # gute Antworten bis "absolviert" (Tier 1)
|
||||||
MASTERY = 10 # Score bis "verstanden" (Tier 2)
|
MASTERY = 10 # Score bis "verstanden" (Tier 2)
|
||||||
|
MEISTERN = 25 # Score bis "gemeistert" (Tier 3, Maximum)
|
||||||
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:
|
def score_berechnen(
|
||||||
|
score_vor_frage: int, gut: bool, tier2: bool, tier3: bool, absolviert: bool, gemeistert: bool,
|
||||||
|
) -> int:
|
||||||
"""Neuer Score nach einer Antwort · driftfrei (immer aus dem Basis-Score gerechnet).
|
"""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).
|
Drei Stufen, freigeschaltet über Guide-Flags:
|
||||||
Tier 2 (tier2=True): +1 bei richtig (Deckel MASTERY=10), −1 bei falsch.
|
- Tier 1 (tier2=False): +1 bei richtig, KEINE Strafe, Deckel NOETIG (3).
|
||||||
Boden ist NOETIG, sobald der Baustein absolviert ist — sonst 0 (kann nicht
|
- Tier 2 (tier2, nicht tier3): +1 / −1, Boden 3, Deckel MASTERY (10).
|
||||||
unter 3 fallen, absolviert bleibt erhalten). Re-Bewertung nutzt denselben
|
- Tier 3 (tier3, Meisterpfad): +1 / −2, Boden 10, Deckel MEISTERN (25).
|
||||||
Basis-Score und ersetzt so das vorige Ergebnis (kein Doppelzählen).
|
Boden vor dem Absolvieren ist 0 (sonst NOETIG — absolviert bleibt erhalten).
|
||||||
|
Ist der Baustein gemeistert, friert der Score bei MEISTERN ein (keine Punkte mehr).
|
||||||
|
Re-Bewertung nutzt denselben Basis-Score und ersetzt das vorige Ergebnis.
|
||||||
"""
|
"""
|
||||||
delta = 1 if gut else (-1 if tier2 else 0)
|
if gemeistert:
|
||||||
floor = NOETIG if absolviert else 0
|
return MEISTERN
|
||||||
cap = MASTERY if tier2 else NOETIG
|
if not tier2:
|
||||||
|
delta, floor, cap = (1 if gut else 0), (NOETIG if absolviert else 0), NOETIG
|
||||||
|
elif not tier3:
|
||||||
|
delta, floor, cap = (1 if gut else -1), NOETIG, MASTERY
|
||||||
|
else:
|
||||||
|
delta, floor, cap = (1 if gut else -2), MASTERY, MEISTERN
|
||||||
return max(floor, min(cap, score_vor_frage + delta))
|
return max(floor, min(cap, score_vor_frage + delta))
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -195,7 +195,8 @@ class BausteinPruefungRequest(BaseModel):
|
|||||||
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)
|
||||||
score_vor_frage: int = 0 # Score, als die Frage gestellt wurde → driftfreies (Re-)Bewerten
|
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
|
tier2: bool = False # ganzer Guide absolviert (alle ≥3) → −1 bei falsch, Deckel 10
|
||||||
|
tier3: bool = False # ganzer Guide verstanden (alle ≥10) → Meisterpfad, −2 bei falsch, Deckel 25
|
||||||
messages: list[ChatMessage] = [] # Dialog bisher; leer = erste Frage
|
messages: list[ChatMessage] = [] # Dialog bisher; leer = erste Frage
|
||||||
provider: ProviderType = "claude"
|
provider: ProviderType = "claude"
|
||||||
|
|
||||||
@@ -208,12 +209,14 @@ class BausteinPruefungResponse(BaseModel):
|
|||||||
gute_antworten: int
|
gute_antworten: int
|
||||||
absolviert: bool
|
absolviert: bool
|
||||||
verstanden: bool = False
|
verstanden: bool = False
|
||||||
|
gemeistert: bool = False
|
||||||
|
|
||||||
|
|
||||||
class BausteinLernstand(BaseModel):
|
class BausteinLernstand(BaseModel):
|
||||||
gute_antworten: int
|
gute_antworten: int
|
||||||
absolviert: bool
|
absolviert: bool
|
||||||
verstanden: bool
|
verstanden: bool
|
||||||
|
gemeistert: bool
|
||||||
vertiefung: bool
|
vertiefung: bool
|
||||||
deepdive: bool
|
deepdive: bool
|
||||||
|
|
||||||
|
|||||||
@@ -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, set_baustein_score, set_baustein_absolviert, set_baustein_verstanden, delete_baustein_daten,
|
list_baustein_progress, set_baustein_score, set_baustein_absolviert, set_baustein_verstanden, set_baustein_gemeistert, 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, MASTERY, baustein_chat, baustein_diskussion, baustein_element_anlegen, pruefung_bewertung, pruefung_frage, score_berechnen, vertiefung_generieren
|
from lernen import NOETIG, MASTERY, MEISTERN, 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
|
||||||
@@ -162,6 +162,7 @@ async def baustein_lernstand(topic: str):
|
|||||||
"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,
|
"verstanden": p["verstanden"] is not None,
|
||||||
|
"gemeistert": p["gemeistert"] 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()),
|
||||||
}
|
}
|
||||||
@@ -170,7 +171,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, "verstanden": False,
|
"gute_antworten": 0, "absolviert": False, "verstanden": False, "gemeistert": False,
|
||||||
"vertiefung": "vertiefung" in arten, "deepdive": "deepdive" in arten,
|
"vertiefung": "vertiefung" in arten, "deepdive": "deepdive" in arten,
|
||||||
}
|
}
|
||||||
return {"bausteine": bausteine}
|
return {"bausteine": bausteine}
|
||||||
@@ -209,11 +210,12 @@ 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, "verstanden": None},
|
{"gute_antworten": 0, "absolviert": None, "verstanden": None, "gemeistert": 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
|
verstanden = stand["verstanden"] is not None
|
||||||
|
gemeistert = stand["gemeistert"] 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]
|
||||||
|
|
||||||
@@ -221,7 +223,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, "verstanden": verstanden}
|
return {"frage": frage, "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden, "gemeistert": gemeistert}
|
||||||
|
|
||||||
if req.aktion == "diskussion":
|
if req.aktion == "diskussion":
|
||||||
if not req.frage.strip():
|
if not req.frage.strip():
|
||||||
@@ -232,7 +234,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, "verstanden": verstanden}
|
return {"reply": reply, "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden, "gemeistert": gemeistert}
|
||||||
|
|
||||||
# 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).
|
||||||
@@ -247,7 +249,9 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
|
|||||||
raise HTTPException(502, "Bewertung fehlgeschlagen — bitte erneut versuchen")
|
raise HTTPException(502, "Bewertung fehlgeschlagen — bitte erneut versuchen")
|
||||||
|
|
||||||
# Score driftfrei aus dem Basis-Score rechnen (Re-Bewertung ersetzt das vorige Ergebnis).
|
# Score driftfrei aus dem Basis-Score rechnen (Re-Bewertung ersetzt das vorige Ergebnis).
|
||||||
score = score_berechnen(req.score_vor_frage, data["bewertung"] == "gut", req.tier2, absolviert)
|
score = score_berechnen(
|
||||||
|
req.score_vor_frage, data["bewertung"] == "gut", req.tier2, req.tier3, absolviert, gemeistert,
|
||||||
|
)
|
||||||
gute = await set_baustein_score(req.topic, req.baustein, score)
|
gute = await set_baustein_score(req.topic, req.baustein, score)
|
||||||
if score >= NOETIG and not absolviert:
|
if score >= NOETIG and not absolviert:
|
||||||
absolviert = True
|
absolviert = True
|
||||||
@@ -256,7 +260,10 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
|
|||||||
if score >= MASTERY and not verstanden:
|
if score >= MASTERY and not verstanden:
|
||||||
await set_baustein_verstanden(req.topic, req.baustein)
|
await set_baustein_verstanden(req.topic, req.baustein)
|
||||||
verstanden = True
|
verstanden = True
|
||||||
return {"feedback": data["feedback"], "bewertung": data["bewertung"], "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden}
|
if score >= MEISTERN and not gemeistert:
|
||||||
|
await set_baustein_gemeistert(req.topic, req.baustein)
|
||||||
|
gemeistert = True
|
||||||
|
return {"feedback": data["feedback"], "bewertung": data["bewertung"], "gute_antworten": gute, "absolviert": absolviert, "verstanden": verstanden, "gemeistert": gemeistert}
|
||||||
|
|
||||||
|
|
||||||
# --- Guides ---
|
# --- Guides ---
|
||||||
|
|||||||
@@ -9,15 +9,17 @@ 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, verstanden, vertiefung}
|
status: { type: Object, default: null }, // {gute_antworten, absolviert, verstanden, gemeistert, vertiefung}
|
||||||
tier2: { type: Boolean, default: false }, // Mastery frei (ganzer Guide absolviert)
|
tier2: { type: Boolean, default: false }, // Tier 2 frei (ganzer Guide absolviert)
|
||||||
|
tier3: { type: Boolean, default: false }, // Meisterpfad frei (ganzer Guide verstanden)
|
||||||
})
|
})
|
||||||
|
|
||||||
const emit = defineEmits(['statusChanged'])
|
const emit = defineEmits(['statusChanged'])
|
||||||
|
|
||||||
const NOETIG = 3 // absolviert
|
const NOETIG = 3 // absolviert
|
||||||
const MAX = 10 // verstanden
|
const MAX = 10 // verstanden
|
||||||
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, verstanden: false, vertiefung: false, deepdive: false })
|
const MEISTERN = 25 // gemeistert (Maximum)
|
||||||
|
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, verstanden: false, gemeistert: 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'
|
||||||
@@ -88,7 +90,7 @@ function onPruefScroll() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function applyPruefung(res) {
|
function applyPruefung(res) {
|
||||||
emit('statusChanged', { ...st.value, gute_antworten: res.gute_antworten, absolviert: res.absolviert, verstanden: res.verstanden })
|
emit('statusChanged', { ...st.value, gute_antworten: res.gute_antworten, absolviert: res.absolviert, verstanden: res.verstanden, gemeistert: res.gemeistert })
|
||||||
}
|
}
|
||||||
|
|
||||||
async function pruefScroll() {
|
async function pruefScroll() {
|
||||||
@@ -158,12 +160,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, score_vor_frage: scoreVorFrage.value, tier2: props.tier2 }, bewerten)
|
pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, score_vor_frage: scoreVorFrage.value, tier2: props.tier2, tier3: props.tier3 }, bewerten)
|
||||||
}
|
}
|
||||||
|
|
||||||
function neuBewerten() {
|
function neuBewerten() {
|
||||||
if (pruefLoading.value) return
|
if (pruefLoading.value) return
|
||||||
pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, score_vor_frage: scoreVorFrage.value, tier2: props.tier2 }, bewerten)
|
pruefSenden({ aktion: 'antwort', frage: aktuelleFrage.value, score_vor_frage: scoreVorFrage.value, tier2: props.tier2, tier3: props.tier3 }, bewerten)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -181,7 +183,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.verstanden" class="bp-chip gold" title="Verstanden (10/10)">✓✓</span>
|
<span v-if="st.gemeistert" class="bp-chip gold" title="Gemeistert (25/25)">✓✓✓ Max</span>
|
||||||
|
<span v-else-if="st.verstanden && tier3" class="bp-chip lila" title="Meisterpfad">✓✓ {{ st.gute_antworten }}/{{ MEISTERN }}</span>
|
||||||
|
<span v-else-if="st.verstanden" class="bp-chip lila" 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 && 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.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>
|
||||||
@@ -232,7 +236,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.verstanden">✓✓ Verstanden ({{ st.gute_antworten }}/{{ MAX }}) — Gold bleibt dir.</template>
|
<template v-if="st.gemeistert">✓✓✓ Gemeistert ({{ MEISTERN }}/{{ MEISTERN }}) — Max. Du kannst dich weiter prüfen, ohne Punkte.</template>
|
||||||
|
<template v-else-if="st.verstanden && tier3">Meisterpfad: {{ st.gute_antworten }}/{{ MEISTERN }}. Richtig = +1, falsch = −2 (nicht unter {{ MAX }}). Bei {{ MEISTERN }} gemeistert.</template>
|
||||||
|
<template v-else-if="st.verstanden">✓✓ Verstanden. Der Meisterpfad ({{ MAX }}→{{ MEISTERN }}) öffnet, sobald der ganze Guide verstanden ist.</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 && 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-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>
|
||||||
@@ -306,6 +312,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.lila { background: color-mix(in srgb, #8b5cf6 16%, var(--panel)); border-color: #8b5cf6; color: #6d28d9; }
|
||||||
.bp-chip.gold { background: color-mix(in srgb, #d4af37 20%, var(--panel)); border-color: #d4af37; color: #8a6d12; }
|
.bp-chip.gold { background: color-mix(in srgb, #d4af37 20%, var(--panel)); border-color: #d4af37; color: #8a6d12; }
|
||||||
|
|
||||||
.bp-panel {
|
.bp-panel {
|
||||||
|
|||||||
@@ -55,11 +55,16 @@ 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.
|
// Tier 2 (Score 3→10) frei, sobald ALLE Bausteine absolviert; Tier 3 (Meisterpfad 10→25) frei, sobald ALLE verstanden.
|
||||||
|
const guideSections = computed(() => (content.value?.chapters || []).flatMap((ch) => ch.sections))
|
||||||
const guideAbsolviert = computed(() => {
|
const guideAbsolviert = computed(() => {
|
||||||
const secs = (content.value?.chapters || []).flatMap((ch) => ch.sections)
|
const secs = guideSections.value
|
||||||
return secs.length > 0 && secs.every((s) => lernstand.value[s.title]?.absolviert)
|
return secs.length > 0 && secs.every((s) => lernstand.value[s.title]?.absolviert)
|
||||||
})
|
})
|
||||||
|
const guideVerstanden = computed(() => {
|
||||||
|
const secs = guideSections.value
|
||||||
|
return secs.length > 0 && secs.every((s) => lernstand.value[s.title]?.verstanden)
|
||||||
|
})
|
||||||
|
|
||||||
// --- Chat (Mechanik in useChat; Kontext-Extraktion bleibt hier) ---
|
// --- Chat (Mechanik in useChat; Kontext-Extraktion bleibt hier) ---
|
||||||
const chat = useChat((msgs) => {
|
const chat = useChat((msgs) => {
|
||||||
@@ -159,12 +164,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]?.verstanden ? 'verstanden' : (lernstand[s.title]?.absolviert ? 'absolviert' : '')]"
|
:class="['section-card', isOnePager && s.key ? 'op-card op-' + s.key : '', lernstand[s.title]?.gemeistert ? 'gemeistert' : (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]?.verstanden" class="baustein-done verstanden" title="Vollständig verstanden (10/10)">✓✓ Verstanden</span>
|
<span v-if="lernstand[s.title]?.gemeistert" class="baustein-done gemeistert" title="Gemeistert (25/25)">✓✓✓ Gemeistert</span>
|
||||||
|
<span v-else-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>
|
<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>
|
||||||
@@ -176,6 +182,7 @@ function extractContext() {
|
|||||||
:provider="provider"
|
:provider="provider"
|
||||||
:status="lernstand[s.title]"
|
:status="lernstand[s.title]"
|
||||||
:tier2="guideAbsolviert"
|
:tier2="guideAbsolviert"
|
||||||
|
:tier3="guideVerstanden"
|
||||||
@status-changed="(st) => onBausteinStatus(s.title, st)"
|
@status-changed="(st) => onBausteinStatus(s.title, st)"
|
||||||
/>
|
/>
|
||||||
</article>
|
</article>
|
||||||
@@ -333,16 +340,28 @@ 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 */
|
/* Verstandene Bausteine (10/10): Lila */
|
||||||
.baustein-done.verstanden {
|
.baustein-done.verstanden {
|
||||||
background: color-mix(in srgb, #d4af37 18%, var(--panel));
|
background: color-mix(in srgb, #8b5cf6 16%, var(--panel));
|
||||||
|
border-color: #8b5cf6;
|
||||||
|
color: #6d28d9;
|
||||||
|
}
|
||||||
|
.guide-content:not(.onepager) .section-card.verstanden {
|
||||||
|
border-color: #8b5cf6;
|
||||||
|
border-top: 3px solid #8b5cf6;
|
||||||
|
background: color-mix(in srgb, #8b5cf6 7%, var(--panel));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Gemeisterte Bausteine (Meisterpfad 25/25): Gold */
|
||||||
|
.baustein-done.gemeistert {
|
||||||
|
background: color-mix(in srgb, #d4af37 20%, var(--panel));
|
||||||
border-color: #d4af37;
|
border-color: #d4af37;
|
||||||
color: #8a6d12;
|
color: #8a6d12;
|
||||||
}
|
}
|
||||||
.guide-content:not(.onepager) .section-card.verstanden {
|
.guide-content:not(.onepager) .section-card.gemeistert {
|
||||||
border-color: #d4af37;
|
border-color: #d4af37;
|
||||||
border-top: 3px solid #d4af37;
|
border-top: 3px solid #d4af37;
|
||||||
background: color-mix(in srgb, #d4af37 7%, var(--panel));
|
background: color-mix(in srgb, #d4af37 8%, 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) */
|
||||||
|
|||||||
Reference in New Issue
Block a user