update
This commit is contained in:
@@ -47,14 +47,15 @@ CREATE TABLE IF NOT EXISTS elements (
|
|||||||
)
|
)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
CREATE_VERTIEFUNGEN = """
|
CREATE_BAUSTEIN_TEXTE = """
|
||||||
CREATE TABLE IF NOT EXISTS vertiefungen (
|
CREATE TABLE IF NOT EXISTS baustein_texte (
|
||||||
topic TEXT NOT NULL,
|
topic TEXT NOT NULL,
|
||||||
baustein TEXT NOT NULL,
|
baustein TEXT NOT NULL,
|
||||||
|
art TEXT NOT NULL,
|
||||||
md TEXT NOT NULL,
|
md TEXT NOT NULL,
|
||||||
created_at TEXT NOT NULL,
|
created_at TEXT NOT NULL,
|
||||||
updated_at TEXT NOT NULL,
|
updated_at TEXT NOT NULL,
|
||||||
PRIMARY KEY (topic, baustein)
|
PRIMARY KEY (topic, baustein, art)
|
||||||
)
|
)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -89,12 +90,20 @@ async def init_db():
|
|||||||
await db.execute(CREATE_PROGRESS)
|
await db.execute(CREATE_PROGRESS)
|
||||||
await db.execute(CREATE_TOPICS)
|
await db.execute(CREATE_TOPICS)
|
||||||
await db.execute(CREATE_ELEMENTS)
|
await db.execute(CREATE_ELEMENTS)
|
||||||
await db.execute(CREATE_VERTIEFUNGEN)
|
await db.execute(CREATE_BAUSTEIN_TEXTE)
|
||||||
await db.execute(CREATE_BAUSTEIN_PROGRESS)
|
await db.execute(CREATE_BAUSTEIN_PROGRESS)
|
||||||
try: # Migration für Bestands-DBs ohne step-Spalte
|
try: # Migration für Bestands-DBs ohne step-Spalte
|
||||||
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
|
||||||
|
# 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'")
|
||||||
|
if await cursor.fetchone():
|
||||||
|
await db.execute(
|
||||||
|
"INSERT OR IGNORE INTO baustein_texte (topic, baustein, art, md, created_at, updated_at) "
|
||||||
|
"SELECT topic, baustein, 'deepdive', md, created_at, updated_at FROM vertiefungen"
|
||||||
|
)
|
||||||
|
await db.execute("DROP TABLE vertiefungen")
|
||||||
await db.execute(
|
await db.execute(
|
||||||
"UPDATE guides SET status = 'error', progress = NULL, error_msg = 'Server-Neustart' "
|
"UPDATE guides SET status = 'error', progress = NULL, error_msg = 'Server-Neustart' "
|
||||||
"WHERE status IN ('queued', 'generating')"
|
"WHERE status IN ('queued', 'generating')"
|
||||||
@@ -288,33 +297,37 @@ def _now() -> str:
|
|||||||
return datetime.now(timezone.utc).isoformat()
|
return datetime.now(timezone.utc).isoformat()
|
||||||
|
|
||||||
|
|
||||||
async def get_vertiefung(topic: str, baustein: str) -> str | None:
|
async def get_vertiefung(topic: str, baustein: str, art: str) -> str | None:
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
cursor = await db.execute(
|
cursor = await db.execute(
|
||||||
"SELECT md FROM vertiefungen WHERE topic = ? AND baustein = ?", (topic, baustein)
|
"SELECT md FROM baustein_texte WHERE topic = ? AND baustein = ? AND art = ?",
|
||||||
|
(topic, baustein, art),
|
||||||
)
|
)
|
||||||
row = await cursor.fetchone()
|
row = await cursor.fetchone()
|
||||||
return row[0] if row else None
|
return row[0] if row else None
|
||||||
|
|
||||||
|
|
||||||
async def set_vertiefung(topic: str, baustein: str, md: str) -> None:
|
async def set_vertiefung(topic: str, baustein: str, art: str, md: str) -> None:
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
now = _now()
|
now = _now()
|
||||||
await db.execute(
|
await db.execute(
|
||||||
"""INSERT INTO vertiefungen (topic, baustein, md, created_at, updated_at)
|
"""INSERT INTO baustein_texte (topic, baustein, art, md, created_at, updated_at)
|
||||||
VALUES (?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?)
|
||||||
ON CONFLICT(topic, baustein) DO UPDATE SET md = excluded.md, updated_at = excluded.updated_at""",
|
ON CONFLICT(topic, baustein, art) DO UPDATE SET md = excluded.md, updated_at = excluded.updated_at""",
|
||||||
(topic, baustein, md, now, now),
|
(topic, baustein, art, md, now, now),
|
||||||
)
|
)
|
||||||
await db.commit()
|
await db.commit()
|
||||||
|
|
||||||
|
|
||||||
async def list_vertiefungen(topic: str) -> set[str]:
|
async def list_vertiefungen(topic: str) -> dict[str, set[str]]:
|
||||||
"""Baustein-Titel, zu denen eine Vertiefung existiert."""
|
"""Baustein-Titel → vorhandene Text-Arten ('vertiefung'/'deepdive')."""
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
cursor = await db.execute("SELECT baustein FROM vertiefungen WHERE topic = ?", (topic,))
|
cursor = await db.execute("SELECT baustein, art FROM baustein_texte WHERE topic = ?", (topic,))
|
||||||
rows = await cursor.fetchall()
|
rows = await cursor.fetchall()
|
||||||
return {row[0] for row in rows}
|
out: dict[str, set[str]] = {}
|
||||||
|
for baustein, art in rows:
|
||||||
|
out.setdefault(baustein, set()).add(art)
|
||||||
|
return out
|
||||||
|
|
||||||
|
|
||||||
async def list_baustein_progress(topic: str) -> list[dict]:
|
async def list_baustein_progress(topic: str) -> list[dict]:
|
||||||
@@ -377,6 +390,6 @@ async def list_baustein_absolviert_all() -> dict[str, set[str]]:
|
|||||||
|
|
||||||
async def delete_baustein_daten(topic: str) -> None:
|
async def delete_baustein_daten(topic: str) -> None:
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
await db.execute("DELETE FROM vertiefungen WHERE topic = ?", (topic,))
|
await db.execute("DELETE FROM baustein_texte WHERE topic = ?", (topic,))
|
||||||
await db.execute("DELETE FROM baustein_progress WHERE topic = ?", (topic,))
|
await db.execute("DELETE FROM baustein_progress WHERE topic = ?", (topic,))
|
||||||
await db.commit()
|
await db.commit()
|
||||||
|
|||||||
@@ -31,11 +31,14 @@ def _transcript(messages: list[dict]) -> str:
|
|||||||
) or "(leer)"
|
) or "(leer)"
|
||||||
|
|
||||||
|
|
||||||
async def vertiefung_generieren(topic: str, baustein: str, section: str, provider: str = DEFAULT_PROVIDER) -> str | None:
|
async def vertiefung_generieren(topic: str, baustein: str, section: str, art: str = "vertiefung", provider: str = DEFAULT_PROVIDER) -> str | None:
|
||||||
"""Ausführliche Fassung des Bausteins als Markdown · None bei Fehler."""
|
"""Ausführlichere Fassung des Bausteins als Markdown · None bei Fehler.
|
||||||
|
|
||||||
|
art "vertiefung" = kurz (~75–150 Wörter), "deepdive" = lang (300–600 Wörter).
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
prompt = _prompt(
|
prompt = _prompt(
|
||||||
"Baustein-Vertiefung",
|
"Baustein-Deepdive" if art == "deepdive" else "Baustein-Vertiefung",
|
||||||
topic=topic, baustein=baustein,
|
topic=topic, baustein=baustein,
|
||||||
section_block=section.strip() or "(keine Guide-Fassung übergeben)",
|
section_block=section.strip() or "(keine Guide-Fassung übergeben)",
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -160,10 +160,14 @@ class ProgressResponse(BaseModel):
|
|||||||
|
|
||||||
# --- Baustein-Lernen ---
|
# --- Baustein-Lernen ---
|
||||||
|
|
||||||
|
VertiefungArt = Literal["vertiefung", "deepdive"]
|
||||||
|
|
||||||
|
|
||||||
class VertiefungRequest(BaseModel):
|
class VertiefungRequest(BaseModel):
|
||||||
topic: str = Field(min_length=1, max_length=100)
|
topic: str = Field(min_length=1, max_length=100)
|
||||||
baustein: str = Field(min_length=1, max_length=200)
|
baustein: str = Field(min_length=1, max_length=200)
|
||||||
section: str = Field(default="", max_length=20000)
|
section: str = Field(default="", max_length=20000)
|
||||||
|
art: VertiefungArt = "vertiefung"
|
||||||
provider: ProviderType = "claude"
|
provider: ProviderType = "claude"
|
||||||
|
|
||||||
|
|
||||||
@@ -202,6 +206,7 @@ class BausteinLernstand(BaseModel):
|
|||||||
gute_antworten: int
|
gute_antworten: int
|
||||||
absolviert: bool
|
absolviert: bool
|
||||||
vertiefung: bool
|
vertiefung: bool
|
||||||
|
deepdive: bool
|
||||||
|
|
||||||
|
|
||||||
class BausteinLernstandResponse(BaseModel):
|
class BausteinLernstandResponse(BaseModel):
|
||||||
|
|||||||
@@ -147,44 +147,56 @@ async def remove_bausteine(topic: str):
|
|||||||
|
|
||||||
# --- Baustein-Lernen: Vertiefung, Chat, Prüfung ---
|
# --- Baustein-Lernen: Vertiefung, Chat, Prüfung ---
|
||||||
|
|
||||||
|
async def _bester_text(topic: str, baustein: str) -> str | None:
|
||||||
|
"""Kontext für Chat/Prüfung: Deep Dive bevorzugt, sonst kurze Vertiefung."""
|
||||||
|
return await get_vertiefung(topic, baustein, "deepdive") or await get_vertiefung(topic, baustein, "vertiefung")
|
||||||
|
|
||||||
|
|
||||||
@router.get("/bausteine/lernstand", response_model=BausteinLernstandResponse)
|
@router.get("/bausteine/lernstand", response_model=BausteinLernstandResponse)
|
||||||
async def baustein_lernstand(topic: str):
|
async def baustein_lernstand(topic: str):
|
||||||
"""Prüfungs-Stand + Vertiefungs-Existenz pro Baustein (roher Titel als Key)."""
|
"""Prüfungs-Stand + Vertiefungs-/Deepdive-Existenz pro Baustein (roher Titel als Key)."""
|
||||||
progress = await list_baustein_progress(topic)
|
progress = await list_baustein_progress(topic)
|
||||||
mit_vertiefung = await list_vertiefungen(topic)
|
texte = await list_vertiefungen(topic)
|
||||||
bausteine = {
|
bausteine = {
|
||||||
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,
|
||||||
"vertiefung": p["baustein"] in mit_vertiefung,
|
"vertiefung": "vertiefung" in texte.get(p["baustein"], set()),
|
||||||
|
"deepdive": "deepdive" in texte.get(p["baustein"], set()),
|
||||||
}
|
}
|
||||||
for p in progress
|
for p in progress
|
||||||
}
|
}
|
||||||
for b in mit_vertiefung - set(bausteine):
|
for b, arten in texte.items():
|
||||||
bausteine[b] = {"gute_antworten": 0, "absolviert": False, "vertiefung": True}
|
if b not in bausteine:
|
||||||
|
bausteine[b] = {
|
||||||
|
"gute_antworten": 0, "absolviert": False,
|
||||||
|
"vertiefung": "vertiefung" in arten, "deepdive": "deepdive" in arten,
|
||||||
|
}
|
||||||
return {"bausteine": bausteine}
|
return {"bausteine": bausteine}
|
||||||
|
|
||||||
|
|
||||||
@router.get("/bausteine/vertiefung", response_model=VertiefungResponse)
|
@router.get("/bausteine/vertiefung", response_model=VertiefungResponse)
|
||||||
async def get_baustein_vertiefung(topic: str, baustein: str):
|
async def get_baustein_vertiefung(topic: str, baustein: str, art: str = "vertiefung"):
|
||||||
md = await get_vertiefung(topic, baustein)
|
if art not in ("vertiefung", "deepdive"):
|
||||||
|
raise HTTPException(400, "Unbekannte Art")
|
||||||
|
md = await get_vertiefung(topic, baustein, art)
|
||||||
if md is None:
|
if md is None:
|
||||||
raise HTTPException(404, "Keine Vertiefung vorhanden")
|
raise HTTPException(404, "Kein Text vorhanden")
|
||||||
return {"md": md}
|
return {"md": md}
|
||||||
|
|
||||||
|
|
||||||
@router.post("/bausteine/vertiefung", response_model=VertiefungResponse)
|
@router.post("/bausteine/vertiefung", response_model=VertiefungResponse)
|
||||||
async def create_baustein_vertiefung(req: VertiefungRequest):
|
async def create_baustein_vertiefung(req: VertiefungRequest):
|
||||||
md = await vertiefung_generieren(req.topic, req.baustein, req.section, provider=req.provider)
|
md = await vertiefung_generieren(req.topic, req.baustein, req.section, art=req.art, provider=req.provider)
|
||||||
if md is None:
|
if md is None:
|
||||||
raise HTTPException(502, "Vertiefung fehlgeschlagen — bitte erneut versuchen")
|
raise HTTPException(502, "Generierung fehlgeschlagen — bitte erneut versuchen")
|
||||||
await set_vertiefung(req.topic, req.baustein, md)
|
await set_vertiefung(req.topic, req.baustein, req.art, md)
|
||||||
return {"md": md}
|
return {"md": md}
|
||||||
|
|
||||||
|
|
||||||
@router.post("/bausteine/chat", response_model=BausteinChatResponse)
|
@router.post("/bausteine/chat", response_model=BausteinChatResponse)
|
||||||
async def baustein_chat_route(req: BausteinChatRequest):
|
async def baustein_chat_route(req: BausteinChatRequest):
|
||||||
vertiefung = await get_vertiefung(req.topic, req.baustein)
|
vertiefung = await _bester_text(req.topic, req.baustein)
|
||||||
reply = await baustein_chat(
|
reply = await baustein_chat(
|
||||||
req.topic, req.baustein, req.section, vertiefung,
|
req.topic, req.baustein, req.section, vertiefung,
|
||||||
[m.model_dump() for m in req.messages], provider=req.provider,
|
[m.model_dump() for m in req.messages], provider=req.provider,
|
||||||
@@ -198,7 +210,7 @@ async def baustein_pruefung_route(req: BausteinPruefungRequest):
|
|||||||
(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},
|
||||||
)
|
)
|
||||||
vertiefung = await get_vertiefung(req.topic, req.baustein)
|
vertiefung = await _bester_text(req.topic, req.baustein)
|
||||||
data = await baustein_pruefung(
|
data = await baustein_pruefung(
|
||||||
req.topic, req.baustein, req.section, vertiefung,
|
req.topic, req.baustein, req.section, vertiefung,
|
||||||
[m.model_dump() for m in req.messages], stand["gute_antworten"], provider=req.provider,
|
[m.model_dump() for m in req.messages], stand["gute_antworten"], provider=req.provider,
|
||||||
|
|||||||
@@ -66,18 +66,18 @@ export async function fetchBausteinLernstand(topic) {
|
|||||||
return jsonOrThrow(res)
|
return jsonOrThrow(res)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchVertiefung(topic, baustein) {
|
export async function fetchVertiefung(topic, baustein, art = 'vertiefung') {
|
||||||
const res = await fetch(
|
const res = await fetch(
|
||||||
`${BASE}/bausteine/vertiefung?topic=${encodeURIComponent(topic)}&baustein=${encodeURIComponent(baustein)}`
|
`${BASE}/bausteine/vertiefung?topic=${encodeURIComponent(topic)}&baustein=${encodeURIComponent(baustein)}&art=${art}`
|
||||||
)
|
)
|
||||||
return jsonOrThrow(res)
|
return jsonOrThrow(res)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function createVertiefung({ topic, baustein, section, provider }) {
|
export async function createVertiefung({ topic, baustein, section, art = 'vertiefung', provider }) {
|
||||||
const res = await fetch(`${BASE}/bausteine/vertiefung`, {
|
const res = await fetch(`${BASE}/bausteine/vertiefung`, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({ topic, baustein, section, provider }),
|
body: JSON.stringify({ topic, baustein, section, art, provider }),
|
||||||
})
|
})
|
||||||
return jsonOrThrow(res)
|
return jsonOrThrow(res)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,47 +15,50 @@ const props = defineProps({
|
|||||||
const emit = defineEmits(['statusChanged'])
|
const emit = defineEmits(['statusChanged'])
|
||||||
|
|
||||||
const NOETIG = 3
|
const NOETIG = 3
|
||||||
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, vertiefung: false })
|
const st = computed(() => props.status || { gute_antworten: 0, absolviert: false, vertiefung: false, deepdive: false })
|
||||||
|
|
||||||
// --- Toggle-Bereich ---
|
// --- Toggle-Bereich ---
|
||||||
const activeTab = ref(null) // null | 'vertiefung' | 'chat' | 'pruefung'
|
const activeTab = ref(null) // null | 'vertiefung' | 'deepdive' | 'chat' | 'pruefung'
|
||||||
|
|
||||||
function toggle(tab) {
|
function toggle(tab) {
|
||||||
activeTab.value = activeTab.value === tab ? null : tab
|
activeTab.value = activeTab.value === tab ? null : tab
|
||||||
if (activeTab.value === 'vertiefung') openVertiefung()
|
if (activeTab.value === 'vertiefung' || activeTab.value === 'deepdive') openText(activeTab.value)
|
||||||
if (activeTab.value === 'pruefung') startPruefung()
|
if (activeTab.value === 'pruefung') startPruefung()
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Vertiefung (persistiert) ---
|
// --- Vertiefung (kurz) + Deep Dive (lang), beide persistiert ---
|
||||||
const vert = ref(null)
|
const texte = ref({
|
||||||
const vertLoading = ref(false)
|
vertiefung: { md: null, loading: false, error: '' },
|
||||||
const vertError = ref('')
|
deepdive: { md: null, loading: false, error: '' },
|
||||||
|
})
|
||||||
|
|
||||||
async function openVertiefung() {
|
async function openText(art) {
|
||||||
if (vert.value !== null || vertLoading.value || !st.value.vertiefung) return
|
const t = texte.value[art]
|
||||||
vertLoading.value = true
|
if (t.md !== null || t.loading || !st.value[art]) return
|
||||||
vertError.value = ''
|
t.loading = true
|
||||||
|
t.error = ''
|
||||||
try {
|
try {
|
||||||
vert.value = (await fetchVertiefung(props.topic, props.baustein)).md
|
t.md = (await fetchVertiefung(props.topic, props.baustein, art)).md
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
vertError.value = e.message
|
t.error = e.message
|
||||||
} finally {
|
} finally {
|
||||||
vertLoading.value = false
|
t.loading = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function generateVertiefung() {
|
async function generateText(art) {
|
||||||
vertLoading.value = true
|
const t = texte.value[art]
|
||||||
vertError.value = ''
|
t.loading = true
|
||||||
|
t.error = ''
|
||||||
try {
|
try {
|
||||||
vert.value = (await createVertiefung({
|
t.md = (await createVertiefung({
|
||||||
topic: props.topic, baustein: props.baustein, section: props.section, provider: props.provider,
|
topic: props.topic, baustein: props.baustein, section: props.section, art, provider: props.provider,
|
||||||
})).md
|
})).md
|
||||||
emit('statusChanged', { ...st.value, vertiefung: true })
|
emit('statusChanged', { ...st.value, [art]: true })
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
vertError.value = e.message
|
t.error = e.message
|
||||||
} finally {
|
} finally {
|
||||||
vertLoading.value = false
|
t.loading = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,6 +112,9 @@ async function startPruefung() {
|
|||||||
<button :class="{ active: activeTab === 'vertiefung' }" @click="toggle('vertiefung')">
|
<button :class="{ active: activeTab === 'vertiefung' }" @click="toggle('vertiefung')">
|
||||||
Vertiefung
|
Vertiefung
|
||||||
</button>
|
</button>
|
||||||
|
<button :class="{ active: activeTab === 'deepdive' }" @click="toggle('deepdive')">
|
||||||
|
Deep Dive
|
||||||
|
</button>
|
||||||
<button :class="{ active: activeTab === 'chat' }" @click="toggle('chat')">
|
<button :class="{ active: activeTab === 'chat' }" @click="toggle('chat')">
|
||||||
Chat
|
Chat
|
||||||
</button>
|
</button>
|
||||||
@@ -120,18 +126,20 @@ async function startPruefung() {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="activeTab" class="bp-panel">
|
<div v-if="activeTab" class="bp-panel">
|
||||||
<!-- Vertiefung -->
|
<!-- Vertiefung (kurz) / Deep Dive (lang) -->
|
||||||
<div v-if="activeTab === 'vertiefung'">
|
<div v-if="activeTab === 'vertiefung' || activeTab === 'deepdive'">
|
||||||
<p v-if="vertLoading" class="bp-hint">{{ vert === null ? 'Generiere Vertiefung…' : 'Lade…' }}</p>
|
<p v-if="texte[activeTab].loading" class="bp-hint">{{ texte[activeTab].md === null ? 'Generiere…' : 'Lade…' }}</p>
|
||||||
<template v-else-if="vert">
|
<template v-else-if="texte[activeTab].md">
|
||||||
<div class="markdown" v-html="renderMarkdown(vert)"></div>
|
<div class="markdown" v-html="renderMarkdown(texte[activeTab].md)"></div>
|
||||||
<button class="bp-action" @click="generateVertiefung">Neu generieren</button>
|
<button class="bp-action" @click="generateText(activeTab)">Neu generieren</button>
|
||||||
</template>
|
</template>
|
||||||
<template v-else>
|
<template v-else>
|
||||||
<p class="bp-hint">Noch keine Vertiefung zu diesem Baustein.</p>
|
<p class="bp-hint">{{ activeTab === 'deepdive' ? 'Noch kein Deep Dive zu diesem Baustein.' : 'Noch keine Vertiefung zu diesem Baustein.' }}</p>
|
||||||
<button class="bp-action" @click="generateVertiefung">Vertiefung generieren</button>
|
<button class="bp-action" @click="generateText(activeTab)">
|
||||||
|
{{ activeTab === 'deepdive' ? 'Deep Dive generieren' : 'Vertiefung generieren' }}
|
||||||
|
</button>
|
||||||
</template>
|
</template>
|
||||||
<p v-if="vertError" class="bp-error">{{ vertError }}</p>
|
<p v-if="texte[activeTab].error" class="bp-error">{{ texte[activeTab].error }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Bausteinchat -->
|
<!-- Bausteinchat -->
|
||||||
|
|||||||
19
templates/Prompt/Baustein-Deepdive.md
Normal file
19
templates/Prompt/Baustein-Deepdive.md
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
Schreibe einen Deep Dive zum Baustein "{baustein}" aus dem Lern-Guide zum Thema "{topic}". Der Leser kennt die kompakte Fassung und will tief einsteigen.
|
||||||
|
|
||||||
|
KOMPAKTE FASSUNG AUS DEM GUIDE:
|
||||||
|
{section_block}
|
||||||
|
|
||||||
|
Inhalt des Deep Dives:
|
||||||
|
- Erkläre das Konzept gründlicher: das Warum hinter den Regeln, nicht nur das Wie.
|
||||||
|
- Mehr und reichere Beispiele als im Guide — Varianten, Grenzfälle, ein realistischer Anwendungsfall.
|
||||||
|
- Typische Fehler und Missverständnisse, jeweils mit Korrektur.
|
||||||
|
- Abgrenzung zu verwandten Konzepten, wo Verwechslungsgefahr besteht.
|
||||||
|
- Baue auf der kompakten Fassung auf, wiederhole sie nicht.
|
||||||
|
|
||||||
|
Stil:
|
||||||
|
- Klares Deutsch, direkt, praxisorientiert. Fachbegriffe beim ersten Auftreten kurz erklären.
|
||||||
|
- Markdown: Absätze, `inline-code`, Codeblöcke mit Sprachangabe nur für Code, ### für Zwischenüberschriften, **fett** sparsam.
|
||||||
|
- Umfang: deutlich ausführlicher als die Guide-Fassung, aber kein Roman — etwa 300–600 Wörter plus Beispiele.
|
||||||
|
- KEINE Zusammenfassung am Ende, kein Fazit-Abschnitt.
|
||||||
|
|
||||||
|
Gib NUR das Markdown des Deep Dives aus — keine Einleitung wie "Hier ist…", kein Code-Fence um das Ganze, kein JSON.
|
||||||
@@ -1,18 +1,17 @@
|
|||||||
Schreibe eine Vertiefung zum Baustein "{baustein}" aus dem Lern-Guide zum Thema "{topic}". Der Leser kennt die kompakte Fassung und will tiefer einsteigen.
|
Schreibe eine kurze Vertiefung zum Baustein "{baustein}" aus dem Lern-Guide zum Thema "{topic}". Der Leser kennt die kompakte Fassung und will einen Schritt weiter — nicht mehr.
|
||||||
|
|
||||||
KOMPAKTE FASSUNG AUS DEM GUIDE:
|
KOMPAKTE FASSUNG AUS DEM GUIDE:
|
||||||
{section_block}
|
{section_block}
|
||||||
|
|
||||||
Inhalt der Vertiefung:
|
Inhalt:
|
||||||
- Erkläre das Konzept gründlicher: das Warum hinter den Regeln, nicht nur das Wie.
|
- Erkläre kurz das Warum hinter dem Konzept — den einen Punkt, der im Guide fehlt.
|
||||||
- Mehr und reichere Beispiele als im Guide — Varianten, Grenzfälle, ein realistischer Anwendungsfall.
|
- 1–2 zusätzliche Beispiele: eine andere Variante oder ein Grenzfall, nicht die Guide-Beispiele wiederholen.
|
||||||
- Typische Fehler und Missverständnisse, jeweils mit Korrektur.
|
- Höchstens ein typischer Fehler, wenn er wirklich häufig ist.
|
||||||
- Abgrenzung zu verwandten Konzepten, wo Verwechslungsgefahr besteht.
|
- Nichts wiederholen, was die kompakte Fassung schon sagt.
|
||||||
- Baue auf der kompakten Fassung auf, wiederhole sie nicht.
|
|
||||||
|
|
||||||
Stil:
|
Stil:
|
||||||
- Klares Deutsch, direkt, praxisorientiert. Fachbegriffe beim ersten Auftreten kurz erklären.
|
- Klares Deutsch, direkt. ~75–150 Wörter Prosa plus die Beispiele.
|
||||||
- Markdown: Absätze, `inline-code`, Codeblöcke mit Sprachangabe nur für Code, ### für Zwischenüberschriften, **fett** sparsam.
|
- Markdown: Absätze, `inline-code`, Codeblöcke mit Sprachangabe nur für Code.
|
||||||
- Umfang: deutlich ausführlicher als die Guide-Fassung, aber kein Roman — etwa 300–600 Wörter plus Beispiele.
|
- KEINE Zwischenüberschriften, KEINE Zusammenfassung, kein Fazit.
|
||||||
|
|
||||||
Gib NUR das Markdown der Vertiefung aus — keine Einleitung wie "Hier ist…", kein Code-Fence um das Ganze, kein JSON.
|
Gib NUR das Markdown der Vertiefung aus — keine Einleitung wie "Hier ist…", kein Code-Fence um das Ganze, kein JSON.
|
||||||
|
|||||||
Reference in New Issue
Block a user