update
This commit is contained in:
@@ -2,6 +2,8 @@
|
|||||||
**/__pycache__
|
**/__pycache__
|
||||||
**/*.pyc
|
**/*.pyc
|
||||||
frontend/dist
|
frontend/dist
|
||||||
creator.db
|
storage
|
||||||
|
projects
|
||||||
|
guides.db
|
||||||
.git
|
.git
|
||||||
.env
|
.claude-data
|
||||||
|
|||||||
7
.env.example
Normal file
7
.env.example
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
# Datei nach .env kopieren (wird nicht committet).
|
||||||
|
|
||||||
|
# Claude-Provider: lokal einmal 'claude setup-token' ausführen, Token eintragen.
|
||||||
|
CLAUDE_CODE_OAUTH_TOKEN=
|
||||||
|
|
||||||
|
# MiniMax-Provider: API-Key aus der MiniMax-Console (Coding-Plan).
|
||||||
|
MINIMAX_API_KEY=
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,6 +1,9 @@
|
|||||||
|
storage/
|
||||||
|
projects/
|
||||||
creator.db
|
creator.db
|
||||||
node_modules/
|
node_modules/
|
||||||
frontend/dist/
|
frontend/dist/
|
||||||
__pycache__/
|
__pycache__/
|
||||||
*.pyc
|
*.pyc
|
||||||
|
.claude-data/
|
||||||
.env
|
.env
|
||||||
|
|||||||
12
Dockerfile
12
Dockerfile
@@ -9,13 +9,25 @@ RUN npm run build
|
|||||||
# Stage 2: Runtime
|
# Stage 2: Runtime
|
||||||
FROM python:3.12-slim
|
FROM python:3.12-slim
|
||||||
|
|
||||||
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
|
curl \
|
||||||
|
ca-certificates \
|
||||||
|
gnupg \
|
||||||
|
&& curl -fsSL https://deb.nodesource.com/setup_20.x | bash - \
|
||||||
|
&& apt-get install -y nodejs \
|
||||||
|
&& npm install -g @anthropic-ai/claude-code opencode-ai \
|
||||||
|
&& pip install --no-cache-dir uv \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
RUN useradd -m -u 1000 app
|
RUN useradd -m -u 1000 app
|
||||||
|
|
||||||
COPY backend/requirements.txt /app/backend/requirements.txt
|
COPY backend/requirements.txt /app/backend/requirements.txt
|
||||||
RUN pip install --no-cache-dir -r /app/backend/requirements.txt
|
RUN pip install --no-cache-dir -r /app/backend/requirements.txt
|
||||||
|
|
||||||
COPY --chown=app:app backend/ /app/backend/
|
COPY --chown=app:app backend/ /app/backend/
|
||||||
|
COPY --chown=app:app templates/ /app/templates/
|
||||||
COPY --chown=app:app --from=frontend /build/dist /app/frontend/dist
|
COPY --chown=app:app --from=frontend /build/dist /app/frontend/dist
|
||||||
|
COPY --chown=app:app dev-ops/opencode.json /home/app/.config/opencode/opencode.json
|
||||||
|
|
||||||
RUN chown app:app /app
|
RUN chown app:app /app
|
||||||
|
|
||||||
|
|||||||
38
Makefile
38
Makefile
@@ -1,19 +1,36 @@
|
|||||||
.PHONY: install dev prod stop logs
|
.PHONY: install dev prod stop logs remove auth sync
|
||||||
|
|
||||||
COMPOSE = docker compose
|
COMPOSE = docker compose
|
||||||
|
|
||||||
|
auth:
|
||||||
|
@mkdir -p .claude-data storage projects
|
||||||
|
@chown -R 1000:1000 .claude-data storage projects
|
||||||
|
@grep -q "^CLAUDE_CODE_OAUTH_TOKEN=.\+" .env 2>/dev/null \
|
||||||
|
|| echo "WARNUNG: CLAUDE_CODE_OAUTH_TOKEN fehlt in .env (Claude-Provider inaktiv)."
|
||||||
|
@grep -q "^MINIMAX_API_KEY=.\+" .env 2>/dev/null \
|
||||||
|
|| echo "WARNUNG: MINIMAX_API_KEY fehlt in .env (MiniMax-Provider inaktiv)."
|
||||||
|
@echo "Verzeichnisse angelegt und auf uid 1000 chowned."
|
||||||
|
|
||||||
install:
|
install:
|
||||||
pip install --break-system-packages -r backend/requirements.txt
|
pip install --break-system-packages fastapi uvicorn[standard] aiosqlite uv
|
||||||
cd frontend && npm install
|
cd frontend && npm install
|
||||||
|
npm install -g opencode-ai
|
||||||
|
@mkdir -p $(HOME)/.config/opencode
|
||||||
|
@ln -sfn $(CURDIR)/dev-ops/opencode.json $(HOME)/.config/opencode/opencode.json
|
||||||
|
@if grep -q "^MINIMAX_API_KEY=.\+" .env 2>/dev/null; then \
|
||||||
|
echo "OpenCode-Config verlinkt. MINIMAX_API_KEY aus .env wird von 'make dev' geladen."; \
|
||||||
|
else \
|
||||||
|
echo "OpenCode-Config verlinkt. MINIMAX_API_KEY noch in .env eintragen."; \
|
||||||
|
fi
|
||||||
|
|
||||||
dev:
|
dev:
|
||||||
@echo "Backend: http://localhost:8000"
|
@echo "Backend: http://localhost:8000"
|
||||||
@echo "Frontend: http://localhost:5173"
|
@echo "Frontend: http://localhost:5173"
|
||||||
@cd backend && uvicorn main:app --reload --port 8000 &
|
@set -a; [ -f .env ] && . ./.env; set +a; \
|
||||||
|
cd backend && uvicorn main:app --reload --port 8000 &
|
||||||
@cd frontend && npx vite --port 5173
|
@cd frontend && npx vite --port 5173
|
||||||
|
|
||||||
prod:
|
prod: auth
|
||||||
@touch creator.db
|
|
||||||
$(COMPOSE) up -d --build
|
$(COMPOSE) up -d --build
|
||||||
|
|
||||||
stop:
|
stop:
|
||||||
@@ -24,3 +41,14 @@ stop:
|
|||||||
|
|
||||||
logs:
|
logs:
|
||||||
$(COMPOSE) logs -f
|
$(COMPOSE) logs -f
|
||||||
|
|
||||||
|
remove: stop
|
||||||
|
@echo "Lösche Datenbank und generierte Dateien..."
|
||||||
|
rm -rf storage/*
|
||||||
|
@echo "Fertig."
|
||||||
|
|
||||||
|
sync:
|
||||||
|
@mkdir -p storage/html
|
||||||
|
rsync -avz --progress root@178.104.67.87:/var/www/creator/storage/creator.db storage/
|
||||||
|
rsync -avz --progress --delete root@178.104.67.87:/var/www/creator/storage/html/ storage/html/
|
||||||
|
@echo "Sync abgeschlossen."
|
||||||
|
|||||||
18
README.md
Normal file
18
README.md
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
Bausteine finden
|
||||||
|
- 1+ Agenten suchen Bausteine zum Thema
|
||||||
|
- Baustein bekommt die Einstufung kern, wichtig und rest
|
||||||
|
|
||||||
|
MiniGuide generieren
|
||||||
|
- 1 Agent erstellt den Guide
|
||||||
|
- Nur Themen (kern) verwenden
|
||||||
|
- MiniGuide-Format bestimmt den Stil
|
||||||
|
|
||||||
|
Guide generieren
|
||||||
|
- 1 Agent erstellt den Guide
|
||||||
|
- Nur Themen (kern+wichtig) verwenden
|
||||||
|
- Guide-Format bestimmt den Stil
|
||||||
|
|
||||||
|
Flow
|
||||||
|
- Bausteine finden
|
||||||
|
- HTML erstellen
|
||||||
|
- Code prüfen
|
||||||
139
backend/agents.py
Normal file
139
backend/agents.py
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
"""Provider-Schicht: führt Agent-Aufrufe über die Claude-CLI oder OpenCode (MiniMax) aus.
|
||||||
|
|
||||||
|
Beide Runner sind unabhängig. Fehlt ein Binary/Key, schlägt nur der
|
||||||
|
jeweilige Provider fehl — der andere läuft unverändert weiter.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from config import PROVIDERS, DEFAULT_PROVIDER
|
||||||
|
|
||||||
|
_active_processes: dict[str, asyncio.subprocess.Process] = {}
|
||||||
|
|
||||||
|
# Capability → Claude --allowedTools
|
||||||
|
_CLAUDE_TOOLS = {
|
||||||
|
"full": "Write,Bash,Read,WebSearch,WebFetch",
|
||||||
|
"files": "Read,Bash,Write",
|
||||||
|
"read": "Read",
|
||||||
|
"none": None,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Capability → OpenCode-Agent (Tool-Rechte in dev-ops/opencode.json definiert)
|
||||||
|
_OPENCODE_AGENTS = {
|
||||||
|
"full": "full",
|
||||||
|
"files": "files",
|
||||||
|
"read": "readonly",
|
||||||
|
"none": "text",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def provider_available(provider: str) -> bool:
|
||||||
|
cfg = PROVIDERS.get(provider)
|
||||||
|
if not cfg:
|
||||||
|
return False
|
||||||
|
if shutil.which(cfg["cli"]) is None:
|
||||||
|
return False
|
||||||
|
env_key = cfg.get("env_key")
|
||||||
|
if env_key and not os.environ.get(env_key):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def kill_process(agent_key: str) -> None:
|
||||||
|
process = _active_processes.get(agent_key)
|
||||||
|
if process and process.returncode is None:
|
||||||
|
process.kill()
|
||||||
|
|
||||||
|
|
||||||
|
async def run_agent(
|
||||||
|
agent_key: str,
|
||||||
|
prompt: str,
|
||||||
|
timeout: int,
|
||||||
|
provider: str = DEFAULT_PROVIDER,
|
||||||
|
role: str = "fast",
|
||||||
|
capabilities: str = "none",
|
||||||
|
) -> tuple[int, str, str]:
|
||||||
|
if provider not in PROVIDERS:
|
||||||
|
return 1, "", f"Unbekannter Provider: {provider}"
|
||||||
|
if shutil.which(PROVIDERS[provider]["cli"]) is None:
|
||||||
|
return 1, "", f"CLI '{PROVIDERS[provider]['cli']}' nicht installiert (Provider: {provider})"
|
||||||
|
timeout = int(timeout * PROVIDERS[provider].get("timeout_factor", 1))
|
||||||
|
if provider == "minimax":
|
||||||
|
return await _run_opencode(agent_key, prompt, timeout, role, capabilities)
|
||||||
|
return await _run_claude_cli(agent_key, prompt, timeout, role, capabilities)
|
||||||
|
|
||||||
|
|
||||||
|
async def _communicate(agent_key: str, cmd: list[str], stdin_data: bytes | None, timeout: int) -> tuple[int, str, str]:
|
||||||
|
process = await asyncio.create_subprocess_exec(
|
||||||
|
*cmd,
|
||||||
|
stdin=asyncio.subprocess.PIPE if stdin_data is not None else asyncio.subprocess.DEVNULL,
|
||||||
|
stdout=asyncio.subprocess.PIPE,
|
||||||
|
stderr=asyncio.subprocess.PIPE,
|
||||||
|
)
|
||||||
|
_active_processes[agent_key] = process
|
||||||
|
try:
|
||||||
|
try:
|
||||||
|
stdout, stderr = await asyncio.wait_for(
|
||||||
|
process.communicate(input=stdin_data),
|
||||||
|
timeout=timeout,
|
||||||
|
)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
process.kill()
|
||||||
|
try:
|
||||||
|
await asyncio.wait_for(process.wait(), timeout=5)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
pass
|
||||||
|
raise
|
||||||
|
return process.returncode, stdout.decode("utf-8", errors="replace"), stderr.decode("utf-8", errors="replace")
|
||||||
|
finally:
|
||||||
|
_active_processes.pop(agent_key, None)
|
||||||
|
|
||||||
|
|
||||||
|
async def _run_claude_cli(agent_key: str, prompt: str, timeout: int, role: str, capabilities: str) -> tuple[int, str, str]:
|
||||||
|
cfg = PROVIDERS["claude"]
|
||||||
|
cmd = [cfg["cli"], "-p", "--model", cfg[role]]
|
||||||
|
tools = _CLAUDE_TOOLS.get(capabilities)
|
||||||
|
if tools:
|
||||||
|
cmd += ["--allowedTools", tools]
|
||||||
|
cmd += ["--dangerously-skip-permissions"]
|
||||||
|
return await _communicate(agent_key, cmd, prompt.encode("utf-8"), timeout)
|
||||||
|
|
||||||
|
|
||||||
|
async def _run_opencode(agent_key: str, prompt: str, timeout: int, role: str, capabilities: str) -> tuple[int, str, str]:
|
||||||
|
cfg = PROVIDERS["minimax"]
|
||||||
|
# Prompt über Tempdatei statt argv (ARG_MAX-Schutz bei großen Projekt-Prompts)
|
||||||
|
with tempfile.NamedTemporaryFile("w", suffix=".md", delete=False, encoding="utf-8", dir=tempfile.gettempdir()) as f:
|
||||||
|
f.write(prompt)
|
||||||
|
prompt_path = Path(f.name)
|
||||||
|
# Positional-Message MUSS vor -f stehen: -f ist ein Array-Flag und
|
||||||
|
# frisst sonst den Text als zweiten Dateinamen ("File not found").
|
||||||
|
cmd = [
|
||||||
|
cfg["cli"], "run",
|
||||||
|
"Folge exakt den Anweisungen in der angehängten Datei. Sie sind der vollständige Auftrag.",
|
||||||
|
"-m", cfg[role],
|
||||||
|
"--agent", _OPENCODE_AGENTS.get(capabilities, "text"),
|
||||||
|
"--dangerously-skip-permissions",
|
||||||
|
"-f", str(prompt_path),
|
||||||
|
]
|
||||||
|
try:
|
||||||
|
rc, stdout, stderr = await _communicate(agent_key, cmd, None, timeout)
|
||||||
|
return rc, _clean_opencode_output(stdout), stderr
|
||||||
|
finally:
|
||||||
|
prompt_path.unlink(missing_ok=True)
|
||||||
|
|
||||||
|
|
||||||
|
_ANSI_RE = re.compile(r"\x1b\[[0-9;]*m")
|
||||||
|
|
||||||
|
|
||||||
|
def _clean_opencode_output(text: str) -> str:
|
||||||
|
"""Entfernt ANSI-Codes und den führenden Banner ("> agent · modell")."""
|
||||||
|
text = _ANSI_RE.sub("", text)
|
||||||
|
lines = text.splitlines()
|
||||||
|
while lines and (not lines[0].strip() or lines[0].lstrip().startswith(">")):
|
||||||
|
lines.pop(0)
|
||||||
|
return "\n".join(lines).strip()
|
||||||
@@ -1,5 +1,38 @@
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
PROJECT_ROOT = Path(__file__).resolve().parent.parent
|
PROJECT_ROOT = Path(__file__).resolve().parent.parent
|
||||||
|
TEMPLATES_DIR = PROJECT_ROOT / "templates"
|
||||||
|
STORAGE_DIR = PROJECT_ROOT / "storage"
|
||||||
FRONTEND_DIST = PROJECT_ROOT / "frontend" / "dist"
|
FRONTEND_DIST = PROJECT_ROOT / "frontend" / "dist"
|
||||||
DB_PATH = PROJECT_ROOT / "creator.db"
|
DB_PATH = STORAGE_DIR / "creator.db"
|
||||||
|
PROJECTS_DIR = PROJECT_ROOT / "projects"
|
||||||
|
|
||||||
|
FORMAT_META = {
|
||||||
|
"OnePager": {"pages": "1 Seite", "time": "~5 Min"},
|
||||||
|
"MiniGuide": {"pages": "3-5 Seiten", "time": "~15-25 Min"},
|
||||||
|
"Guide": {"pages": "10-30 Seiten", "time": "variabel"},
|
||||||
|
}
|
||||||
|
|
||||||
|
AGENT_TIMEOUT = 3600
|
||||||
|
|
||||||
|
MAX_CONCURRENT_GENERATIONS = 10
|
||||||
|
|
||||||
|
# Provider-Stacks: komplett unabhängig, einer kann jederzeit entfernt werden.
|
||||||
|
# Rollen: "guide" = große Generierung, "fast" = Baustein-Recherche/Chat.
|
||||||
|
DEFAULT_PROVIDER = "claude"
|
||||||
|
PROVIDERS = {
|
||||||
|
"claude": {
|
||||||
|
"cli": "claude",
|
||||||
|
"guide": "claude-opus-4-8[1m]",
|
||||||
|
"fast": "claude-sonnet-4-6",
|
||||||
|
"env_key": None, # Auth via CLAUDE_CODE_OAUTH_TOKEN oder ~/.claude
|
||||||
|
"timeout_factor": 1,
|
||||||
|
},
|
||||||
|
"minimax": {
|
||||||
|
"cli": "opencode",
|
||||||
|
"guide": "minimax/MiniMax-M3",
|
||||||
|
"fast": "minimax/MiniMax-M3",
|
||||||
|
"env_key": "MINIMAX_API_KEY",
|
||||||
|
"timeout_factor": 3, # M3 ist bei großen Dokumenten deutlich langsamer
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,7 +1,29 @@
|
|||||||
import aiosqlite
|
import aiosqlite
|
||||||
|
|
||||||
from config import DB_PATH
|
from config import DB_PATH
|
||||||
|
|
||||||
|
CREATE_GUIDES = """
|
||||||
|
CREATE TABLE IF NOT EXISTS guides (
|
||||||
|
id TEXT PRIMARY KEY,
|
||||||
|
topic TEXT NOT NULL,
|
||||||
|
format TEXT NOT NULL,
|
||||||
|
instructions TEXT NOT NULL DEFAULT '',
|
||||||
|
status TEXT NOT NULL DEFAULT 'queued',
|
||||||
|
progress TEXT,
|
||||||
|
error_msg TEXT,
|
||||||
|
created_at TEXT NOT NULL,
|
||||||
|
updated_at TEXT NOT NULL
|
||||||
|
)
|
||||||
|
"""
|
||||||
|
|
||||||
|
CREATE_PROGRESS = """
|
||||||
|
CREATE TABLE IF NOT EXISTS guide_progress (
|
||||||
|
guide_id TEXT NOT NULL,
|
||||||
|
chapter TEXT NOT NULL,
|
||||||
|
created_at TEXT NOT NULL,
|
||||||
|
PRIMARY KEY (guide_id, chapter)
|
||||||
|
)
|
||||||
|
"""
|
||||||
|
|
||||||
_db: aiosqlite.Connection | None = None
|
_db: aiosqlite.Connection | None = None
|
||||||
|
|
||||||
|
|
||||||
@@ -9,12 +31,18 @@ async def get_db() -> aiosqlite.Connection:
|
|||||||
global _db
|
global _db
|
||||||
if _db is None:
|
if _db is None:
|
||||||
_db = await aiosqlite.connect(DB_PATH)
|
_db = await aiosqlite.connect(DB_PATH)
|
||||||
|
_db.row_factory = None
|
||||||
return _db
|
return _db
|
||||||
|
|
||||||
|
|
||||||
async def init_db():
|
async def init_db():
|
||||||
db = await get_db()
|
db = await get_db()
|
||||||
# Tabellen folgen, sobald die ersten Features stehen.
|
await db.execute(CREATE_GUIDES)
|
||||||
|
await db.execute(CREATE_PROGRESS)
|
||||||
|
await db.execute(
|
||||||
|
"UPDATE guides SET status = 'error', progress = NULL, error_msg = 'Server-Neustart' "
|
||||||
|
"WHERE status IN ('queued', 'generating')"
|
||||||
|
)
|
||||||
await db.commit()
|
await db.commit()
|
||||||
|
|
||||||
|
|
||||||
@@ -28,3 +56,77 @@ async def close_db():
|
|||||||
def _row_to_dict(row, cursor):
|
def _row_to_dict(row, cursor):
|
||||||
columns = [d[0] for d in cursor.description]
|
columns = [d[0] for d in cursor.description]
|
||||||
return dict(zip(columns, row))
|
return dict(zip(columns, row))
|
||||||
|
|
||||||
|
|
||||||
|
async def create_guide(guide: dict) -> dict:
|
||||||
|
db = await get_db()
|
||||||
|
await db.execute(
|
||||||
|
"""INSERT INTO guides (id, topic, format, instructions, status, progress, created_at, updated_at)
|
||||||
|
VALUES (:id, :topic, :format, :instructions, :status, :progress, :created_at, :updated_at)""",
|
||||||
|
guide,
|
||||||
|
)
|
||||||
|
await db.commit()
|
||||||
|
return guide
|
||||||
|
|
||||||
|
|
||||||
|
async def get_guide(guide_id: str) -> dict | None:
|
||||||
|
db = await get_db()
|
||||||
|
cursor = await db.execute("SELECT * FROM guides WHERE id = ?", (guide_id,))
|
||||||
|
row = await cursor.fetchone()
|
||||||
|
if row is None:
|
||||||
|
return None
|
||||||
|
return _row_to_dict(row, cursor)
|
||||||
|
|
||||||
|
|
||||||
|
async def list_guides() -> list[dict]:
|
||||||
|
db = await get_db()
|
||||||
|
cursor = await db.execute("SELECT * FROM guides ORDER BY created_at DESC")
|
||||||
|
rows = await cursor.fetchall()
|
||||||
|
return [_row_to_dict(row, cursor) for row in rows]
|
||||||
|
|
||||||
|
|
||||||
|
async def update_guide(guide_id: str, **fields) -> None:
|
||||||
|
sets = ", ".join(f"{k} = :{k}" for k in fields)
|
||||||
|
fields["id"] = guide_id
|
||||||
|
db = await get_db()
|
||||||
|
await db.execute(f"UPDATE guides SET {sets} WHERE id = :id", fields)
|
||||||
|
await db.commit()
|
||||||
|
|
||||||
|
|
||||||
|
async def delete_guide(guide_id: str) -> bool:
|
||||||
|
db = await get_db()
|
||||||
|
cursor = await db.execute("DELETE FROM guides WHERE id = ?", (guide_id,))
|
||||||
|
await db.commit()
|
||||||
|
return cursor.rowcount > 0
|
||||||
|
|
||||||
|
|
||||||
|
# --- Kapitel-Fortschritt ---
|
||||||
|
|
||||||
|
async def list_progress(guide_id: str) -> list[str]:
|
||||||
|
db = await get_db()
|
||||||
|
cursor = await db.execute(
|
||||||
|
"SELECT chapter FROM guide_progress WHERE guide_id = ?", (guide_id,)
|
||||||
|
)
|
||||||
|
rows = await cursor.fetchall()
|
||||||
|
return [row[0] for row in rows]
|
||||||
|
|
||||||
|
|
||||||
|
async def set_progress(guide_id: str, chapter: str, done: bool) -> None:
|
||||||
|
from datetime import datetime, timezone
|
||||||
|
db = await get_db()
|
||||||
|
if done:
|
||||||
|
await db.execute(
|
||||||
|
"INSERT OR IGNORE INTO guide_progress (guide_id, chapter, created_at) VALUES (?, ?, ?)",
|
||||||
|
(guide_id, chapter, datetime.now(timezone.utc).isoformat()),
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
await db.execute(
|
||||||
|
"DELETE FROM guide_progress WHERE guide_id = ? AND chapter = ?", (guide_id, chapter)
|
||||||
|
)
|
||||||
|
await db.commit()
|
||||||
|
|
||||||
|
|
||||||
|
async def delete_progress(guide_id: str) -> None:
|
||||||
|
db = await get_db()
|
||||||
|
await db.execute("DELETE FROM guide_progress WHERE guide_id = ?", (guide_id,))
|
||||||
|
await db.commit()
|
||||||
|
|||||||
178
backend/generator.py
Normal file
178
backend/generator.py
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
import asyncio
|
||||||
|
import uuid
|
||||||
|
from datetime import datetime, timezone
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from agents import run_agent, kill_process
|
||||||
|
from config import (
|
||||||
|
AGENT_TIMEOUT,
|
||||||
|
DEFAULT_PROVIDER,
|
||||||
|
TEMPLATES_DIR,
|
||||||
|
MAX_CONCURRENT_GENERATIONS,
|
||||||
|
)
|
||||||
|
from database import update_guide
|
||||||
|
from paths import final_html_path, project_dir
|
||||||
|
|
||||||
|
_semaphore = asyncio.Semaphore(MAX_CONCURRENT_GENERATIONS)
|
||||||
|
_cancelled: set[str] = set()
|
||||||
|
|
||||||
|
|
||||||
|
async def cancel_guide(guide_id: str) -> bool:
|
||||||
|
_cancelled.add(guide_id)
|
||||||
|
kill_process(guide_id)
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
await update_guide(guide_id, status="error", progress=None, error_msg="Abgebrochen", updated_at=now)
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
async def _set_progress(guide_id: str, progress: str) -> None:
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
await update_guide(guide_id, progress=progress, updated_at=now)
|
||||||
|
|
||||||
|
|
||||||
|
# Welche Baustein-Kategorien jedes Format abdeckt.
|
||||||
|
FORMAT_COVERAGE = {
|
||||||
|
"OnePager": "NUR die KERN-Bausteine, maximal verdichtet",
|
||||||
|
"MiniGuide": "NUR die KERN-Bausteine",
|
||||||
|
"Guide": "die KERN- und WICHTIG-Bausteine",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def _prompt(name: str, **kwargs) -> str:
|
||||||
|
template = (TEMPLATES_DIR / "Prompt" / f"{name}.md").read_text(encoding="utf-8")
|
||||||
|
return template.format(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
def _extra(instructions: str) -> str:
|
||||||
|
return f"\n\nZUSÄTZLICHE ANWEISUNGEN VOM NUTZER:\n{instructions}\n" if instructions else ""
|
||||||
|
|
||||||
|
|
||||||
|
def _build_bausteine_prompt(topic: str, bausteine_path: Path, instructions: str = "", project: Path | None = None) -> str:
|
||||||
|
if project:
|
||||||
|
source = _prompt("Bausteine-Quelle-Projekt", project=project)
|
||||||
|
else:
|
||||||
|
source = _prompt("Bausteine-Quelle-Thema", topic=topic)
|
||||||
|
return _prompt(
|
||||||
|
"Bausteine",
|
||||||
|
topic=topic, source=source, bausteine_path=bausteine_path, extra=_extra(instructions),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _build_guide_prompt(topic: str, format_name: str, html_path: Path, bausteine: str, instructions: str = "", project: Path | None = None) -> str:
|
||||||
|
spec = (TEMPLATES_DIR / "Format" / f"{format_name}.md").read_text(encoding="utf-8")
|
||||||
|
reference = (TEMPLATES_DIR / "Referenz" / f"{format_name}.md").read_text(encoding="utf-8")
|
||||||
|
|
||||||
|
if project:
|
||||||
|
facts = _prompt("Guide-Fakten-Projekt", project=project)
|
||||||
|
else:
|
||||||
|
facts = _prompt("Guide-Fakten-Thema")
|
||||||
|
|
||||||
|
return _prompt(
|
||||||
|
"Guide",
|
||||||
|
topic=topic, format_name=format_name, html_path=html_path,
|
||||||
|
bausteine=bausteine, coverage=FORMAT_COVERAGE[format_name],
|
||||||
|
facts=facts, spec=spec, reference=reference, extra=_extra(instructions),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def generate_guide(guide_id: str, topic: str, format_name: str, instructions: str = "", provider: str = DEFAULT_PROVIDER) -> None:
|
||||||
|
async with _semaphore:
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
await update_guide(guide_id, status="generating", progress="Ermittle Bausteine…", updated_at=now)
|
||||||
|
|
||||||
|
html_path = final_html_path(topic, format_name)
|
||||||
|
bausteine_path = html_path.with_suffix(".bausteine.md")
|
||||||
|
project = project_dir(topic) if project_dir(topic).is_dir() else None
|
||||||
|
|
||||||
|
try:
|
||||||
|
if guide_id in _cancelled:
|
||||||
|
return
|
||||||
|
|
||||||
|
# Step 1: Bausteine ermitteln (Thema: Websuche, Projekt: Dateien lesen)
|
||||||
|
current_step = "Bausteine"
|
||||||
|
bs_prompt = _build_bausteine_prompt(topic, bausteine_path, instructions, project)
|
||||||
|
returncode, bs_out, bs_err = await run_agent(
|
||||||
|
guide_id, bs_prompt, AGENT_TIMEOUT,
|
||||||
|
provider=provider, role="fast", capabilities="files" if project else "full",
|
||||||
|
)
|
||||||
|
|
||||||
|
if guide_id in _cancelled:
|
||||||
|
return
|
||||||
|
if returncode != 0:
|
||||||
|
await _fail(guide_id, _claude_error("Baustein-Fehler", returncode, bs_out, bs_err))
|
||||||
|
return
|
||||||
|
if not bausteine_path.exists():
|
||||||
|
await _fail(guide_id, "Baustein-Datei wurde nicht erstellt")
|
||||||
|
return
|
||||||
|
bausteine = bausteine_path.read_text(encoding="utf-8")
|
||||||
|
|
||||||
|
# Step 2: Generator-Agent erstellt HTML nach Bausteinen
|
||||||
|
await _set_progress(guide_id, "Generiere HTML…")
|
||||||
|
current_step = "Generierung"
|
||||||
|
gen_prompt = _build_guide_prompt(topic, format_name, html_path, bausteine, instructions, project)
|
||||||
|
returncode, stdout, stderr = await run_agent(guide_id, gen_prompt, AGENT_TIMEOUT, provider=provider, role="guide", capabilities="full")
|
||||||
|
|
||||||
|
if guide_id in _cancelled:
|
||||||
|
return
|
||||||
|
if returncode != 0:
|
||||||
|
await _fail(guide_id, _claude_error("Generator-Fehler", returncode, stdout, stderr))
|
||||||
|
return
|
||||||
|
|
||||||
|
if not html_path.exists():
|
||||||
|
await _fail(guide_id, "HTML-Datei wurde nicht erstellt")
|
||||||
|
return
|
||||||
|
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
await update_guide(
|
||||||
|
guide_id, status="done", progress=None, updated_at=now,
|
||||||
|
)
|
||||||
|
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
await _fail(guide_id, f"Timeout bei {current_step} nach {AGENT_TIMEOUT}s")
|
||||||
|
except Exception as e:
|
||||||
|
await _fail(guide_id, str(e)[:2000])
|
||||||
|
finally:
|
||||||
|
_cancelled.discard(guide_id)
|
||||||
|
|
||||||
|
|
||||||
|
def _claude_error(label: str, returncode: int, stdout: str, stderr: str) -> str:
|
||||||
|
stderr = (stderr or "").strip()
|
||||||
|
if stderr:
|
||||||
|
return f"{label}: {stderr[:1000]}"
|
||||||
|
tail = (stdout or "").strip()[-500:]
|
||||||
|
if tail:
|
||||||
|
return f"{label} (exit {returncode}, stderr leer): …{tail}"
|
||||||
|
return f"{label} (exit {returncode}, ohne Ausgabe)"
|
||||||
|
|
||||||
|
|
||||||
|
async def _fail(guide_id: str, msg: str) -> None:
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
await update_guide(guide_id, status="error", progress=None, error_msg=msg, updated_at=now)
|
||||||
|
|
||||||
|
|
||||||
|
def _build_guide_chat_prompt(topic: str, format_name: str, section: str, outline: str, messages: list[dict]) -> str:
|
||||||
|
transcript = "\n".join(
|
||||||
|
f"{'Nutzer' if m.get('role') == 'user' else 'Assistent'}: {m.get('content', '')}"
|
||||||
|
for m in messages
|
||||||
|
)
|
||||||
|
return _prompt(
|
||||||
|
"Chat",
|
||||||
|
topic=topic, format_name=format_name,
|
||||||
|
outline_block=outline.strip() or "(keine)",
|
||||||
|
section_block=section.strip() or "(kein Abschnitt erkannt)",
|
||||||
|
transcript=transcript,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def chat_with_guide(topic: str, format_name: str, section: str, outline: str, messages: list[dict], provider: str = DEFAULT_PROVIDER) -> str:
|
||||||
|
try:
|
||||||
|
prompt = _build_guide_chat_prompt(topic, format_name, section, outline, messages)
|
||||||
|
returncode, stdout, stderr = await run_agent(
|
||||||
|
"chat-" + str(uuid.uuid4()), prompt, 240, provider=provider, role="fast", capabilities="none"
|
||||||
|
)
|
||||||
|
if returncode != 0:
|
||||||
|
return "Entschuldigung, das hat nicht geklappt. Bitte versuche es erneut."
|
||||||
|
reply = stdout.strip()
|
||||||
|
return reply or "Entschuldigung, ich habe keine Antwort erhalten."
|
||||||
|
except Exception:
|
||||||
|
return "Entschuldigung, das hat nicht geklappt. Bitte versuche es erneut."
|
||||||
@@ -3,13 +3,14 @@ from contextlib import asynccontextmanager
|
|||||||
from fastapi import FastAPI
|
from fastapi import FastAPI
|
||||||
from fastapi.staticfiles import StaticFiles
|
from fastapi.staticfiles import StaticFiles
|
||||||
|
|
||||||
from config import FRONTEND_DIST
|
from config import FRONTEND_DIST, STORAGE_DIR
|
||||||
from database import init_db, close_db
|
from database import init_db, close_db
|
||||||
from routes import router
|
from routes import router
|
||||||
|
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def lifespan(app: FastAPI):
|
async def lifespan(app: FastAPI):
|
||||||
|
(STORAGE_DIR / "html").mkdir(parents=True, exist_ok=True)
|
||||||
await init_db()
|
await init_db()
|
||||||
yield
|
yield
|
||||||
await close_db()
|
await close_db()
|
||||||
|
|||||||
62
backend/models.py
Normal file
62
backend/models.py
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
from pydantic import BaseModel, Field
|
||||||
|
from typing import Literal
|
||||||
|
|
||||||
|
FormatType = Literal[
|
||||||
|
"OnePager",
|
||||||
|
"MiniGuide",
|
||||||
|
"Guide",
|
||||||
|
]
|
||||||
|
|
||||||
|
ProviderType = Literal["claude", "minimax"]
|
||||||
|
|
||||||
|
|
||||||
|
class GuideCreateRequest(BaseModel):
|
||||||
|
topic: str = Field(min_length=1, max_length=100)
|
||||||
|
format: FormatType
|
||||||
|
instructions: str = Field(default="", max_length=2000)
|
||||||
|
provider: ProviderType = "claude"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectResponse(BaseModel):
|
||||||
|
name: str
|
||||||
|
|
||||||
|
|
||||||
|
class ProviderInfo(BaseModel):
|
||||||
|
id: str
|
||||||
|
available: bool
|
||||||
|
|
||||||
|
|
||||||
|
class GuideResponse(BaseModel):
|
||||||
|
id: str
|
||||||
|
topic: str
|
||||||
|
format: str
|
||||||
|
status: str
|
||||||
|
progress: str | None = None
|
||||||
|
error_msg: str | None = None
|
||||||
|
created_at: str
|
||||||
|
updated_at: str
|
||||||
|
|
||||||
|
|
||||||
|
class ChatMessage(BaseModel):
|
||||||
|
role: Literal["user", "assistant"]
|
||||||
|
content: str = Field(min_length=1, max_length=8000)
|
||||||
|
|
||||||
|
|
||||||
|
class GuideChatRequest(BaseModel):
|
||||||
|
section: str = Field(default="", max_length=20000)
|
||||||
|
outline: str = Field(default="", max_length=8000)
|
||||||
|
messages: list[ChatMessage] = Field(min_length=1)
|
||||||
|
provider: ProviderType = "claude"
|
||||||
|
|
||||||
|
|
||||||
|
class GuideChatResponse(BaseModel):
|
||||||
|
reply: str
|
||||||
|
|
||||||
|
|
||||||
|
class ProgressUpdate(BaseModel):
|
||||||
|
chapter: str = Field(min_length=1, max_length=100)
|
||||||
|
done: bool
|
||||||
|
|
||||||
|
|
||||||
|
class ProgressResponse(BaseModel):
|
||||||
|
chapters: list[str]
|
||||||
16
backend/paths.py
Normal file
16
backend/paths.py
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from config import STORAGE_DIR, PROJECTS_DIR
|
||||||
|
|
||||||
|
|
||||||
|
def safe_basename(topic: str, format_name: str) -> str:
|
||||||
|
clean = topic.replace("/", "_").replace("\x00", "")
|
||||||
|
return f"{clean} - {format_name}"
|
||||||
|
|
||||||
|
|
||||||
|
def final_html_path(topic: str, format_name: str) -> Path:
|
||||||
|
return STORAGE_DIR / "html" / f"{safe_basename(topic, format_name)}.html"
|
||||||
|
|
||||||
|
|
||||||
|
def project_dir(name: str) -> Path:
|
||||||
|
return PROJECTS_DIR / name
|
||||||
@@ -1,8 +1,151 @@
|
|||||||
from fastapi import APIRouter
|
import asyncio
|
||||||
|
import shutil
|
||||||
|
import uuid
|
||||||
|
from datetime import datetime, timezone
|
||||||
|
|
||||||
|
from fastapi import APIRouter, HTTPException
|
||||||
|
from fastapi.responses import FileResponse
|
||||||
|
|
||||||
|
from agents import provider_available
|
||||||
|
from config import FORMAT_META, PROJECTS_DIR, PROVIDERS
|
||||||
|
from database import (
|
||||||
|
create_guide, delete_guide, get_guide, list_guides,
|
||||||
|
list_progress, set_progress, delete_progress,
|
||||||
|
)
|
||||||
|
from generator import generate_guide, cancel_guide, chat_with_guide
|
||||||
|
from models import (
|
||||||
|
GuideCreateRequest, GuideResponse,
|
||||||
|
GuideChatRequest, GuideChatResponse,
|
||||||
|
ProgressUpdate, ProgressResponse, ProjectResponse, ProviderInfo,
|
||||||
|
)
|
||||||
|
from paths import final_html_path, project_dir
|
||||||
|
|
||||||
router = APIRouter(prefix="/api")
|
router = APIRouter(prefix="/api")
|
||||||
|
|
||||||
|
|
||||||
@router.get("/health")
|
@router.get("/formats")
|
||||||
async def health():
|
async def get_formats():
|
||||||
|
return FORMAT_META
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/providers", response_model=list[ProviderInfo])
|
||||||
|
async def get_providers():
|
||||||
|
return [{"id": pid, "available": provider_available(pid)} for pid in PROVIDERS]
|
||||||
|
|
||||||
|
|
||||||
|
def _safe_project_name(name: str) -> str:
|
||||||
|
if not name or "/" in name or "\\" in name or ".." in name or "\x00" in name:
|
||||||
|
raise HTTPException(400, "Ungültiger Projektname")
|
||||||
|
return name
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/projects", response_model=list[ProjectResponse])
|
||||||
|
async def list_projects():
|
||||||
|
if not PROJECTS_DIR.is_dir():
|
||||||
|
return []
|
||||||
|
return [{"name": entry.name} for entry in sorted(PROJECTS_DIR.iterdir()) if entry.is_dir()]
|
||||||
|
|
||||||
|
|
||||||
|
@router.delete("/projects/{name}")
|
||||||
|
async def remove_project(name: str):
|
||||||
|
_safe_project_name(name)
|
||||||
|
pdir = project_dir(name)
|
||||||
|
if not pdir.is_dir():
|
||||||
|
raise HTTPException(404, "Projekt nicht gefunden")
|
||||||
|
shutil.rmtree(pdir)
|
||||||
return {"ok": True}
|
return {"ok": True}
|
||||||
|
|
||||||
|
|
||||||
|
@router.post("/guides", response_model=GuideResponse)
|
||||||
|
async def create(req: GuideCreateRequest):
|
||||||
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
|
guide = {
|
||||||
|
"id": str(uuid.uuid4()),
|
||||||
|
"topic": req.topic.strip(),
|
||||||
|
"format": req.format,
|
||||||
|
"instructions": req.instructions.strip(),
|
||||||
|
"status": "queued",
|
||||||
|
"progress": None,
|
||||||
|
"created_at": now,
|
||||||
|
"updated_at": now,
|
||||||
|
}
|
||||||
|
await create_guide(guide)
|
||||||
|
asyncio.create_task(generate_guide(guide["id"], guide["topic"], guide["format"], guide["instructions"], req.provider))
|
||||||
|
return guide
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/guides", response_model=list[GuideResponse])
|
||||||
|
async def list_all():
|
||||||
|
return await list_guides()
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/guides/{guide_id}", response_model=GuideResponse)
|
||||||
|
async def get_one(guide_id: str):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
return guide
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/guides/{guide_id}/html")
|
||||||
|
async def download_html(guide_id: str):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
if guide["status"] != "done":
|
||||||
|
raise HTTPException(404, "HTML nicht verfügbar")
|
||||||
|
html_path = final_html_path(guide["topic"], guide["format"])
|
||||||
|
if not html_path.exists():
|
||||||
|
raise HTTPException(404, "Datei nicht gefunden")
|
||||||
|
return FileResponse(html_path, media_type="text/html", content_disposition_type="inline")
|
||||||
|
|
||||||
|
|
||||||
|
@router.post("/guides/{guide_id}/chat", response_model=GuideChatResponse)
|
||||||
|
async def guide_chat(guide_id: str, req: GuideChatRequest):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
reply = await chat_with_guide(
|
||||||
|
guide["topic"], guide["format"], req.section, req.outline,
|
||||||
|
[m.model_dump() for m in req.messages],
|
||||||
|
provider=req.provider,
|
||||||
|
)
|
||||||
|
return {"reply": reply}
|
||||||
|
|
||||||
|
|
||||||
|
@router.post("/guides/{guide_id}/cancel")
|
||||||
|
async def cancel(guide_id: str):
|
||||||
|
cancelled = await cancel_guide(guide_id)
|
||||||
|
if not cancelled:
|
||||||
|
raise HTTPException(404, "Kein aktiver Prozess gefunden")
|
||||||
|
return {"ok": True}
|
||||||
|
|
||||||
|
|
||||||
|
@router.delete("/guides/{guide_id}")
|
||||||
|
async def remove(guide_id: str):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
html_path = final_html_path(guide["topic"], guide["format"])
|
||||||
|
html_path.unlink(missing_ok=True)
|
||||||
|
html_path.with_suffix(".bausteine.md").unlink(missing_ok=True)
|
||||||
|
await delete_progress(guide_id)
|
||||||
|
await delete_guide(guide_id)
|
||||||
|
return {"ok": True}
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/guides/{guide_id}/progress", response_model=ProgressResponse)
|
||||||
|
async def get_progress(guide_id: str):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
return {"chapters": await list_progress(guide_id)}
|
||||||
|
|
||||||
|
|
||||||
|
@router.post("/guides/{guide_id}/progress", response_model=ProgressResponse)
|
||||||
|
async def update_progress(guide_id: str, req: ProgressUpdate):
|
||||||
|
guide = await get_guide(guide_id)
|
||||||
|
if guide is None:
|
||||||
|
raise HTTPException(404, "Guide nicht gefunden")
|
||||||
|
await set_progress(guide_id, req.chapter, req.done)
|
||||||
|
return {"chapters": await list_progress(guide_id)}
|
||||||
|
|||||||
77
dev-ops/opencode.json
Normal file
77
dev-ops/opencode.json
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://opencode.ai/config.json",
|
||||||
|
"provider": {
|
||||||
|
"minimax": {
|
||||||
|
"options": {
|
||||||
|
"apiKey": "{env:MINIMAX_API_KEY}"
|
||||||
|
},
|
||||||
|
"models": {
|
||||||
|
"MiniMax-M3": {
|
||||||
|
"name": "MiniMax M3"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"mcp": {
|
||||||
|
"minimax-search": {
|
||||||
|
"type": "local",
|
||||||
|
"command": ["uvx", "minimax-coding-plan-mcp"],
|
||||||
|
"environment": {
|
||||||
|
"MINIMAX_API_KEY": "{env:MINIMAX_API_KEY}",
|
||||||
|
"MINIMAX_API_HOST": "https://api.minimax.io"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"agent": {
|
||||||
|
"full": {
|
||||||
|
"description": "Alle Tools: Dateien, Bash, Websuche",
|
||||||
|
"permission": {
|
||||||
|
"edit": "allow",
|
||||||
|
"bash": "allow",
|
||||||
|
"webfetch": "allow"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"files": {
|
||||||
|
"description": "Dateien lesen/schreiben + Bash, keine Websuche",
|
||||||
|
"permission": {
|
||||||
|
"edit": "allow",
|
||||||
|
"bash": "allow",
|
||||||
|
"webfetch": "deny"
|
||||||
|
},
|
||||||
|
"tools": {
|
||||||
|
"minimax-search*": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"readonly": {
|
||||||
|
"description": "Nur Dateien lesen",
|
||||||
|
"permission": {
|
||||||
|
"edit": "deny",
|
||||||
|
"bash": "deny",
|
||||||
|
"webfetch": "deny"
|
||||||
|
},
|
||||||
|
"tools": {
|
||||||
|
"write": false,
|
||||||
|
"edit": false,
|
||||||
|
"bash": false,
|
||||||
|
"minimax-search*": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"text": {
|
||||||
|
"description": "Reine Textantwort, keine Tools",
|
||||||
|
"permission": {
|
||||||
|
"edit": "deny",
|
||||||
|
"bash": "deny",
|
||||||
|
"webfetch": "deny"
|
||||||
|
},
|
||||||
|
"tools": {
|
||||||
|
"write": false,
|
||||||
|
"edit": false,
|
||||||
|
"bash": false,
|
||||||
|
"read": false,
|
||||||
|
"glob": false,
|
||||||
|
"grep": false,
|
||||||
|
"minimax-search*": false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,10 +4,15 @@ services:
|
|||||||
context: .
|
context: .
|
||||||
container_name: creator
|
container_name: creator
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
|
environment:
|
||||||
|
- CLAUDE_CODE_OAUTH_TOKEN=${CLAUDE_CODE_OAUTH_TOKEN:-}
|
||||||
|
- MINIMAX_API_KEY=${MINIMAX_API_KEY:-}
|
||||||
networks:
|
networks:
|
||||||
- web
|
- web
|
||||||
volumes:
|
volumes:
|
||||||
- ./creator.db:/app/creator.db
|
- ./storage:/app/storage
|
||||||
|
- ./projects:/app/projects
|
||||||
|
- ./.claude-data:/home/app/.claude
|
||||||
labels:
|
labels:
|
||||||
- "traefik.enable=true"
|
- "traefik.enable=true"
|
||||||
- "traefik.http.routers.creatorapp.rule=Host(`creator.marha.de`)"
|
- "traefik.http.routers.creatorapp.rule=Host(`creator.marha.de`)"
|
||||||
|
|||||||
39
frontend/.gitignore
vendored
Normal file
39
frontend/.gitignore
vendored
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
lerna-debug.log*
|
||||||
|
|
||||||
|
node_modules
|
||||||
|
.DS_Store
|
||||||
|
dist
|
||||||
|
dist-ssr
|
||||||
|
coverage
|
||||||
|
*.local
|
||||||
|
|
||||||
|
# Editor directories and files
|
||||||
|
.vscode/*
|
||||||
|
!.vscode/extensions.json
|
||||||
|
.idea
|
||||||
|
*.suo
|
||||||
|
*.ntvs*
|
||||||
|
*.njsproj
|
||||||
|
*.sln
|
||||||
|
*.sw?
|
||||||
|
|
||||||
|
*.tsbuildinfo
|
||||||
|
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Cypress
|
||||||
|
/cypress/videos/
|
||||||
|
/cypress/screenshots/
|
||||||
|
|
||||||
|
# Vitest
|
||||||
|
__screenshots__/
|
||||||
|
|
||||||
|
# Vite
|
||||||
|
*.timestamp-*-*.mjs
|
||||||
3
frontend/.vscode/extensions.json
vendored
Normal file
3
frontend/.vscode/extensions.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"recommendations": ["Vue.volar"]
|
||||||
|
}
|
||||||
38
frontend/README.md
Normal file
38
frontend/README.md
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
# frontend
|
||||||
|
|
||||||
|
This template should help get you started developing with Vue 3 in Vite.
|
||||||
|
|
||||||
|
## Recommended IDE Setup
|
||||||
|
|
||||||
|
[VS Code](https://code.visualstudio.com/) + [Vue (Official)](https://marketplace.visualstudio.com/items?itemName=Vue.volar) (and disable Vetur).
|
||||||
|
|
||||||
|
## Recommended Browser Setup
|
||||||
|
|
||||||
|
- Chromium-based browsers (Chrome, Edge, Brave, etc.):
|
||||||
|
- [Vue.js devtools](https://chromewebstore.google.com/detail/vuejs-devtools/nhdogjmejiglipccpnnnanhbledajbpd)
|
||||||
|
- [Turn on Custom Object Formatter in Chrome DevTools](http://bit.ly/object-formatters)
|
||||||
|
- Firefox:
|
||||||
|
- [Vue.js devtools](https://addons.mozilla.org/en-US/firefox/addon/vue-js-devtools/)
|
||||||
|
- [Turn on Custom Object Formatter in Firefox DevTools](https://fxdx.dev/firefox-devtools-custom-object-formatters/)
|
||||||
|
|
||||||
|
## Customize configuration
|
||||||
|
|
||||||
|
See [Vite Configuration Reference](https://vite.dev/config/).
|
||||||
|
|
||||||
|
## Project Setup
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm install
|
||||||
|
```
|
||||||
|
|
||||||
|
### Compile and Hot-Reload for Development
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
### Compile and Minify for Production
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run build
|
||||||
|
```
|
||||||
779
frontend/package-lock.json
generated
779
frontend/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -9,6 +9,8 @@
|
|||||||
"preview": "vite preview"
|
"preview": "vite preview"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"dompurify": "^3.4.7",
|
||||||
|
"marked": "^18.0.4",
|
||||||
"vue": "^3.5.32"
|
"vue": "^3.5.32"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|||||||
BIN
frontend/public/favicon.ico
Normal file
BIN
frontend/public/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.2 KiB |
@@ -1,13 +1,42 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { ref, onMounted } from 'vue'
|
import { ref, computed, onMounted, onUnmounted, nextTick } from 'vue'
|
||||||
import { fetchHealth } from './api.js'
|
import { fetchGuides, createGuide as apiCreate, deleteGuide, cancelGuide as apiCancel, fetchProjects, deleteProject as apiDeleteProject, fetchProviders } from './api.js'
|
||||||
|
import TopicSidebar from './components/TopicSidebar.vue'
|
||||||
|
import TopicDetail from './components/TopicDetail.vue'
|
||||||
|
|
||||||
const backendOk = ref(false)
|
const guides = ref([])
|
||||||
|
const projects = ref([])
|
||||||
|
const manualTopics = ref([])
|
||||||
|
const selectedTopic = ref(null)
|
||||||
|
const previewGuide = ref(null)
|
||||||
|
const sidebarPinned = ref(localStorage.getItem('sidebarPinned') !== 'false')
|
||||||
|
const sidebarSticky = ref(false)
|
||||||
const darkMode = ref(
|
const darkMode = ref(
|
||||||
localStorage.getItem('darkMode') === null
|
localStorage.getItem('darkMode') === null
|
||||||
? window.matchMedia('(prefers-color-scheme: dark)').matches
|
? window.matchMedia('(prefers-color-scheme: dark)').matches
|
||||||
: localStorage.getItem('darkMode') === 'true',
|
: localStorage.getItem('darkMode') === 'true',
|
||||||
)
|
)
|
||||||
|
const provider = ref(localStorage.getItem('provider') || 'claude')
|
||||||
|
const providers = ref([])
|
||||||
|
|
||||||
|
function setProvider(id) {
|
||||||
|
provider.value = id
|
||||||
|
localStorage.setItem('provider', id)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadProviders() {
|
||||||
|
try {
|
||||||
|
providers.value = await fetchProviders()
|
||||||
|
const current = providers.value.find((p) => p.id === provider.value)
|
||||||
|
if (current && !current.available) {
|
||||||
|
const fallback = providers.value.find((p) => p.available)
|
||||||
|
if (fallback) setProvider(fallback.id)
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Fehler beim Laden der Provider:', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let pollTimer = null
|
||||||
|
|
||||||
function applyTheme() {
|
function applyTheme() {
|
||||||
document.documentElement.classList.toggle('dark', darkMode.value)
|
document.documentElement.classList.toggle('dark', darkMode.value)
|
||||||
@@ -21,49 +50,286 @@ function toggleDark() {
|
|||||||
|
|
||||||
applyTheme()
|
applyTheme()
|
||||||
|
|
||||||
onMounted(async () => {
|
function toggleSidebarPin() {
|
||||||
try {
|
sidebarPinned.value = !sidebarPinned.value
|
||||||
const res = await fetchHealth()
|
localStorage.setItem('sidebarPinned', sidebarPinned.value ? 'true' : 'false')
|
||||||
backendOk.value = res.ok === true
|
if (sidebarPinned.value) sidebarSticky.value = false
|
||||||
} catch (e) {
|
}
|
||||||
console.error('Backend nicht erreichbar:', e)
|
|
||||||
|
function clickHoverZone() {
|
||||||
|
sidebarSticky.value = !sidebarSticky.value
|
||||||
|
}
|
||||||
|
|
||||||
|
function onSidebarLeave() {
|
||||||
|
if (!sidebarPinned.value) sidebarSticky.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const projectNames = computed(() => projects.value.map((p) => p.name))
|
||||||
|
|
||||||
|
const topics = computed(() => {
|
||||||
|
const isProject = new Set(projectNames.value)
|
||||||
|
const topicDates = {}
|
||||||
|
for (const g of guides.value) {
|
||||||
|
if (isProject.has(g.topic)) continue
|
||||||
|
if (!topicDates[g.topic] || g.created_at > topicDates[g.topic]) {
|
||||||
|
topicDates[g.topic] = g.created_at
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
for (const t of manualTopics.value) {
|
||||||
|
if (isProject.has(t)) continue
|
||||||
|
if (!topicDates[t]) topicDates[t] = new Date().toISOString()
|
||||||
|
}
|
||||||
|
return Object.keys(topicDates).sort((a, b) => topicDates[b].localeCompare(topicDates[a]))
|
||||||
|
})
|
||||||
|
|
||||||
|
const doneByFormat = computed(() => {
|
||||||
|
const map = {}
|
||||||
|
for (const g of guides.value) {
|
||||||
|
if (g.topic !== selectedTopic.value) continue
|
||||||
|
if (g.status !== 'done') continue
|
||||||
|
if (!map[g.format] || g.created_at > map[g.format].created_at) {
|
||||||
|
map[g.format] = g
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map
|
||||||
|
})
|
||||||
|
|
||||||
|
const latestByFormat = computed(() => {
|
||||||
|
const map = {}
|
||||||
|
for (const g of guides.value) {
|
||||||
|
if (g.topic !== selectedTopic.value) continue
|
||||||
|
if (!map[g.format] || g.created_at > map[g.format].created_at) {
|
||||||
|
map[g.format] = g
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map
|
||||||
|
})
|
||||||
|
|
||||||
|
const hasActiveGuides = computed(() =>
|
||||||
|
guides.value.some((g) => g.status === 'queued' || g.status === 'generating'),
|
||||||
|
)
|
||||||
|
|
||||||
|
async function loadGuides() {
|
||||||
|
try {
|
||||||
|
guides.value = await fetchGuides()
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Fehler beim Laden:', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadProjects() {
|
||||||
|
try {
|
||||||
|
projects.value = await fetchProjects()
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Fehler beim Laden der Projekte:', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const FORMAT_ORDER = ['OnePager', 'MiniGuide', 'Guide']
|
||||||
|
|
||||||
|
function autoPreview() {
|
||||||
|
const map = doneByFormat.value
|
||||||
|
for (const f of FORMAT_ORDER) {
|
||||||
|
if (map[f]) {
|
||||||
|
previewGuide.value = map[f]
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
previewGuide.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectTopic(topic) {
|
||||||
|
selectedTopic.value = topic
|
||||||
|
previewGuide.value = null
|
||||||
|
sidebarSticky.value = false
|
||||||
|
nextTick(autoPreview)
|
||||||
|
}
|
||||||
|
|
||||||
|
function createTopic(topic) {
|
||||||
|
if (!manualTopics.value.includes(topic)) {
|
||||||
|
manualTopics.value.push(topic)
|
||||||
|
}
|
||||||
|
selectedTopic.value = topic
|
||||||
|
previewGuide.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleFormatClick({ format, instructions }) {
|
||||||
|
if (!selectedTopic.value) return
|
||||||
|
await apiCreate(selectedTopic.value, format, instructions, provider.value)
|
||||||
|
await loadGuides()
|
||||||
|
startPolling()
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleDeleteProject(name) {
|
||||||
|
await apiDeleteProject(name)
|
||||||
|
if (selectedTopic.value === name) {
|
||||||
|
selectedTopic.value = null
|
||||||
|
previewGuide.value = null
|
||||||
|
}
|
||||||
|
await loadProjects()
|
||||||
|
}
|
||||||
|
|
||||||
|
function handlePreview(guide) {
|
||||||
|
previewGuide.value = guide
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleDeleteGuide(guideId) {
|
||||||
|
await deleteGuide(guideId)
|
||||||
|
if (previewGuide.value?.id === guideId) {
|
||||||
|
previewGuide.value = null
|
||||||
|
}
|
||||||
|
await loadGuides()
|
||||||
|
}
|
||||||
|
|
||||||
|
function startPolling() {
|
||||||
|
stopPolling()
|
||||||
|
pollTimer = setInterval(async () => {
|
||||||
|
await loadGuides()
|
||||||
|
if (!hasActiveGuides.value) stopPolling()
|
||||||
|
}, 3000)
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopPolling() {
|
||||||
|
if (pollTimer) {
|
||||||
|
clearInterval(pollTimer)
|
||||||
|
pollTimer = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleCancel(guideId) {
|
||||||
|
await apiCancel(guideId)
|
||||||
|
await loadGuides()
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleDeleteTopic(topic) {
|
||||||
|
const topicGuides = guides.value.filter((g) => g.topic === topic)
|
||||||
|
for (const g of topicGuides) {
|
||||||
|
await deleteGuide(g.id)
|
||||||
|
}
|
||||||
|
manualTopics.value = manualTopics.value.filter((t) => t !== topic)
|
||||||
|
if (selectedTopic.value === topic) {
|
||||||
|
selectedTopic.value = null
|
||||||
|
previewGuide.value = null
|
||||||
|
}
|
||||||
|
await loadGuides()
|
||||||
|
}
|
||||||
|
|
||||||
|
function onVisibility() {
|
||||||
|
if (document.hidden) {
|
||||||
|
stopPolling()
|
||||||
|
} else {
|
||||||
|
loadGuides()
|
||||||
|
if (hasActiveGuides.value) startPolling()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
await Promise.all([loadGuides(), loadProjects(), loadProviders()])
|
||||||
|
if (!selectedTopic.value && topics.value.length) {
|
||||||
|
selectTopic(topics.value[0])
|
||||||
|
}
|
||||||
|
document.addEventListener('visibilitychange', onVisibility)
|
||||||
|
})
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
stopPolling()
|
||||||
|
document.removeEventListener('visibilitychange', onVisibility)
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="layout">
|
<div class="layout" :class="{ 'sidebar-floating': !sidebarPinned, 'sidebar-open': sidebarSticky }">
|
||||||
<header class="topbar">
|
<div v-if="!sidebarPinned" class="hover-zone" @click="clickHoverZone"></div>
|
||||||
<h1>Creator</h1>
|
<div v-if="!sidebarPinned && sidebarSticky" class="sidebar-backdrop" @click="sidebarSticky = false"></div>
|
||||||
<button class="theme-toggle" @click="toggleDark">{{ darkMode ? '☀' : '☾' }}</button>
|
<TopicSidebar
|
||||||
</header>
|
:topics="topics"
|
||||||
<main class="main">
|
:projects="projectNames"
|
||||||
<p>Backend: <span :class="backendOk ? 'ok' : 'err'">{{ backendOk ? 'verbunden' : 'nicht erreichbar' }}</span></p>
|
:selectedTopic="selectedTopic"
|
||||||
</main>
|
:doneByFormat="doneByFormat"
|
||||||
|
:latestByFormat="latestByFormat"
|
||||||
|
:allGuides="guides"
|
||||||
|
:pinned="sidebarPinned"
|
||||||
|
:dark="darkMode"
|
||||||
|
:provider="provider"
|
||||||
|
:providers="providers"
|
||||||
|
@setProvider="setProvider"
|
||||||
|
@toggleDark="toggleDark"
|
||||||
|
@select="selectTopic"
|
||||||
|
@create="createTopic"
|
||||||
|
@formatClick="handleFormatClick"
|
||||||
|
@deleteTopic="handleDeleteTopic"
|
||||||
|
@deleteProject="handleDeleteProject"
|
||||||
|
@cancelGuide="handleCancel"
|
||||||
|
@deleteGuide="handleDeleteGuide"
|
||||||
|
@preview="handlePreview"
|
||||||
|
@togglePin="toggleSidebarPin"
|
||||||
|
@sidebarLeave="onSidebarLeave"
|
||||||
|
/>
|
||||||
|
<TopicDetail
|
||||||
|
v-if="selectedTopic"
|
||||||
|
:previewGuide="previewGuide"
|
||||||
|
:dark="darkMode"
|
||||||
|
:provider="provider"
|
||||||
|
/>
|
||||||
|
<div v-else class="empty-main">
|
||||||
|
<p>Thema in der Sidebar anlegen oder auswählen.</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
:root {
|
:root {
|
||||||
--bg: #f8f9fb;
|
--bg: #f8f9fb;
|
||||||
|
--bg-preview: #f0f1f4;
|
||||||
--panel: #ffffff;
|
--panel: #ffffff;
|
||||||
|
--panel-soft: #f4f5f7;
|
||||||
--border: #e2e5e9;
|
--border: #e2e5e9;
|
||||||
|
--border-strong: #d8dde3;
|
||||||
--text: #1a1a1a;
|
--text: #1a1a1a;
|
||||||
--text-muted: #4b5563;
|
--text-muted: #4b5563;
|
||||||
|
--text-faint: #9ca3af;
|
||||||
--accent: #6366f1;
|
--accent: #6366f1;
|
||||||
|
--accent-hover: #4f46e5;
|
||||||
|
--accent-soft: #ede9fe;
|
||||||
|
--accent-border: #a5b4fc;
|
||||||
|
--on-accent: #ffffff;
|
||||||
--success: #065f46;
|
--success: #065f46;
|
||||||
|
--success-soft: #d1fae5;
|
||||||
|
--success-soft-hover: #a7f3d0;
|
||||||
|
--success-border: #34d399;
|
||||||
|
--warning: #92400e;
|
||||||
|
--warning-soft: #fef3c7;
|
||||||
|
--warning-border: #fbbf24;
|
||||||
--danger: #991b1b;
|
--danger: #991b1b;
|
||||||
|
--code-bg: #1e2a3a;
|
||||||
|
--code-fg: #e6e6e6;
|
||||||
|
--shadow: rgba(0, 0, 0, 0.12);
|
||||||
}
|
}
|
||||||
|
|
||||||
html.dark {
|
html.dark {
|
||||||
--bg: #15171c;
|
--bg: #15171c;
|
||||||
|
--bg-preview: #0e1014;
|
||||||
--panel: #1c1f26;
|
--panel: #1c1f26;
|
||||||
|
--panel-soft: #23262e;
|
||||||
--border: #2c3038;
|
--border: #2c3038;
|
||||||
|
--border-strong: #3a3f4a;
|
||||||
--text: #e6e8ee;
|
--text: #e6e8ee;
|
||||||
--text-muted: #9aa3b2;
|
--text-muted: #9aa3b2;
|
||||||
|
--text-faint: #6b7280;
|
||||||
--accent: #6366f1;
|
--accent: #6366f1;
|
||||||
|
--accent-hover: #818cf8;
|
||||||
|
--accent-soft: #2a2350;
|
||||||
|
--accent-border: #4f46e5;
|
||||||
|
--on-accent: #ffffff;
|
||||||
--success: #34d399;
|
--success: #34d399;
|
||||||
|
--success-soft: #0f3a2e;
|
||||||
|
--success-soft-hover: #155e45;
|
||||||
|
--success-border: #0f805e;
|
||||||
|
--warning: #fbbf24;
|
||||||
|
--warning-soft: #3a2c0a;
|
||||||
|
--warning-border: #a06a12;
|
||||||
--danger: #f87171;
|
--danger: #f87171;
|
||||||
|
--shadow: rgba(0, 0, 0, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
* {
|
* {
|
||||||
@@ -78,50 +344,66 @@ body {
|
|||||||
color: var(--text);
|
color: var(--text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input,
|
||||||
|
textarea {
|
||||||
|
background: var(--panel);
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
|
||||||
|
input::placeholder,
|
||||||
|
textarea::placeholder {
|
||||||
|
color: var(--text-faint);
|
||||||
|
}
|
||||||
|
|
||||||
.layout {
|
.layout {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
|
||||||
height: 100vh;
|
height: 100vh;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
.topbar {
|
.hover-zone {
|
||||||
|
position: fixed;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
width: 50px;
|
||||||
|
height: 100vh;
|
||||||
|
z-index: 5;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Unsichtbare Fläche hinter der offenen Floating-Sidebar.
|
||||||
|
Tipp/Klick daneben schließt sie — ohne sie gibt es auf Touch keinen Ausweg. */
|
||||||
|
.sidebar-backdrop {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 9;
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layout.sidebar-floating > .sidebar {
|
||||||
|
position: fixed;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
height: 100vh;
|
||||||
|
transform: translateX(-100%);
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
z-index: 10;
|
||||||
|
box-shadow: 0 0 16px var(--shadow);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layout.sidebar-floating .hover-zone:hover ~ .sidebar,
|
||||||
|
.layout.sidebar-floating > .sidebar:hover,
|
||||||
|
.layout.sidebar-floating.sidebar-open > .sidebar {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-main {
|
||||||
|
flex: 1;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: center;
|
||||||
padding: 0.75rem 1.25rem;
|
|
||||||
background: var(--panel);
|
|
||||||
border-bottom: 1px solid var(--border);
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
font-size: 1.1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.theme-toggle {
|
|
||||||
background: none;
|
|
||||||
border: 1px solid var(--border);
|
|
||||||
border-radius: 6px;
|
|
||||||
color: var(--text);
|
|
||||||
cursor: pointer;
|
|
||||||
font-size: 1rem;
|
|
||||||
padding: 0.25rem 0.6rem;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
border-color: var(--accent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.main {
|
|
||||||
flex: 1;
|
|
||||||
padding: 1.25rem;
|
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
|
font-size: 1rem;
|
||||||
.ok {
|
|
||||||
color: var(--success);
|
|
||||||
}
|
|
||||||
|
|
||||||
.err {
|
|
||||||
color: var(--danger);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,6 +1,64 @@
|
|||||||
const BASE = '/api'
|
const BASE = '/api'
|
||||||
|
|
||||||
export async function fetchHealth() {
|
export async function fetchGuides() {
|
||||||
const res = await fetch(`${BASE}/health`)
|
const res = await fetch(`${BASE}/guides`)
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function createGuide(topic, format, instructions = '', provider = 'claude') {
|
||||||
|
const res = await fetch(`${BASE}/guides`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ topic, format, instructions, provider }),
|
||||||
|
})
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchProviders() {
|
||||||
|
const res = await fetch(`${BASE}/providers`)
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchProjects() {
|
||||||
|
const res = await fetch(`${BASE}/projects`)
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function deleteProject(name) {
|
||||||
|
await fetch(`${BASE}/projects/${encodeURIComponent(name)}`, { method: 'DELETE' })
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function cancelGuide(id) {
|
||||||
|
await fetch(`${BASE}/guides/${id}/cancel`, { method: 'POST' })
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function deleteGuide(id) {
|
||||||
|
await fetch(`${BASE}/guides/${id}`, { method: 'DELETE' })
|
||||||
|
}
|
||||||
|
|
||||||
|
export function htmlUrl(id) {
|
||||||
|
return `${BASE}/guides/${id}/html`
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchProgress(id) {
|
||||||
|
const res = await fetch(`${BASE}/guides/${id}/progress`)
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function setProgress(id, chapter, done) {
|
||||||
|
const res = await fetch(`${BASE}/guides/${id}/progress`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ chapter, done }),
|
||||||
|
})
|
||||||
|
return res.json()
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function chatGuide(id, { section, outline, messages, provider = 'claude' }) {
|
||||||
|
const res = await fetch(`${BASE}/guides/${id}/chat`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ section, outline, messages, provider }),
|
||||||
|
})
|
||||||
return res.json()
|
return res.json()
|
||||||
}
|
}
|
||||||
|
|||||||
637
frontend/src/components/TopicDetail.vue
Normal file
637
frontend/src/components/TopicDetail.vue
Normal file
@@ -0,0 +1,637 @@
|
|||||||
|
<script setup>
|
||||||
|
import { computed, ref, watch, nextTick, onMounted, onUnmounted } from 'vue'
|
||||||
|
import { marked } from 'marked'
|
||||||
|
import DOMPurify from 'dompurify'
|
||||||
|
import { htmlUrl, chatGuide, fetchProgress, setProgress } from '../api.js'
|
||||||
|
|
||||||
|
marked.setOptions({ breaks: true, gfm: true })
|
||||||
|
|
||||||
|
function renderMarkdown(text) {
|
||||||
|
return DOMPurify.sanitize(marked.parse(text || ''))
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
previewGuide: { type: Object, default: null },
|
||||||
|
dark: { type: Boolean, default: false },
|
||||||
|
provider: { type: String, default: 'claude' },
|
||||||
|
})
|
||||||
|
|
||||||
|
const LANDSCAPE_FORMATS = ['OnePager']
|
||||||
|
const isLandscape = computed(() => LANDSCAPE_FORMATS.includes(props.previewGuide?.format))
|
||||||
|
|
||||||
|
const frameEl = ref(null)
|
||||||
|
|
||||||
|
// --- Kapitel-Fortschritt ---
|
||||||
|
function onFrameLoad(e) {
|
||||||
|
const doc = e.target.contentDocument
|
||||||
|
if (!doc) return
|
||||||
|
injectStyles(doc)
|
||||||
|
applyIframeTheme(doc)
|
||||||
|
setupProgress(doc)
|
||||||
|
// Klicks im iframe blubbern nicht zum Eltern-document → eigener Listener (same-origin)
|
||||||
|
doc.addEventListener('mousedown', onFrameMouseDown)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Guide-Dokument folgt dem App-Theme über das data-theme-Attribut.
|
||||||
|
// Guides definieren :root-Variablen (--ink, --muted, --line, --bg-soft);
|
||||||
|
// der injizierte Dark-Block (s. injectStyles) greift nur mit gesetztem Attribut.
|
||||||
|
function applyIframeTheme(doc) {
|
||||||
|
if (props.dark) doc.documentElement.setAttribute('data-theme', 'dark')
|
||||||
|
else doc.documentElement.removeAttribute('data-theme')
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(() => props.dark, () => {
|
||||||
|
const doc = frameEl.value?.contentDocument
|
||||||
|
if (doc) applyIframeTheme(doc)
|
||||||
|
})
|
||||||
|
|
||||||
|
function onFrameMouseDown() {
|
||||||
|
if (chatOpen.value) closeChat()
|
||||||
|
}
|
||||||
|
|
||||||
|
function injectStyles(doc) {
|
||||||
|
// Ohne viewport-meta rendert das iframe-Dokument mit Desktop-Breite und greift unsere max-width-Query nicht.
|
||||||
|
if (!doc.querySelector('meta[name="viewport"]')) {
|
||||||
|
const meta = doc.createElement('meta')
|
||||||
|
meta.name = 'viewport'
|
||||||
|
meta.content = 'width=device-width, initial-scale=1'
|
||||||
|
doc.head?.appendChild(meta)
|
||||||
|
}
|
||||||
|
|
||||||
|
const style = doc.createElement('style')
|
||||||
|
const portrait = !isLandscape.value
|
||||||
|
? `html { background: #f0f1f4; }
|
||||||
|
body {
|
||||||
|
zoom: 1.15;
|
||||||
|
max-width: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 2.5rem 3.5rem;
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 1px 8px rgba(0, 0, 0, 0.08);
|
||||||
|
}`
|
||||||
|
: ''
|
||||||
|
// Generischer Dark-Override: greift nur, wenn die App data-theme="dark" setzt.
|
||||||
|
// Entfällt, wenn der Guide eigene Dark-Regeln mitbringt (neuere Templates).
|
||||||
|
const hasOwnDark = (doc.head?.innerHTML || '').includes('data-theme="dark"')
|
||||||
|
const darkOverride = hasOwnDark
|
||||||
|
? ''
|
||||||
|
: `html[data-theme="dark"] {
|
||||||
|
--ink: #e6e8ee;
|
||||||
|
--muted: #9aa3b2;
|
||||||
|
--line: #2c3038;
|
||||||
|
--bg-soft: #23262e;
|
||||||
|
background: #0e1014;
|
||||||
|
}
|
||||||
|
html[data-theme="dark"] body {
|
||||||
|
background: #1c1f26;
|
||||||
|
color: var(--ink, #e6e8ee);
|
||||||
|
}
|
||||||
|
/* Callouts haben hartkodierte helle Hintergründe (#e8f4ea u. ä.) */
|
||||||
|
html[data-theme="dark"] .callout {
|
||||||
|
background: var(--bg-soft, #23262e);
|
||||||
|
}`
|
||||||
|
style.textContent = `@media screen {
|
||||||
|
${portrait}
|
||||||
|
${darkOverride}
|
||||||
|
.ch-toggle {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
margin: 2.5rem 0 0.5rem;
|
||||||
|
padding: 0.9rem 1rem;
|
||||||
|
border: 1.5px dashed var(--line, #c7ccd6);
|
||||||
|
border-radius: 10px;
|
||||||
|
background: var(--bg-soft, #f8f9fb);
|
||||||
|
color: var(--muted, #4b5563);
|
||||||
|
font: 600 0.95rem/1.2 -apple-system, "Segoe UI", Roboto, sans-serif;
|
||||||
|
text-align: center;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.12s;
|
||||||
|
}
|
||||||
|
.ch-toggle:hover { border-color: #6366f1; color: #6366f1; background: transparent; }
|
||||||
|
.ch-toggle.is-done {
|
||||||
|
border-style: solid;
|
||||||
|
border-color: #34d399;
|
||||||
|
background: #d1fae5;
|
||||||
|
color: #065f46;
|
||||||
|
}
|
||||||
|
html[data-theme="dark"] .ch-toggle.is-done {
|
||||||
|
border-color: #0f805e;
|
||||||
|
background: #0f3a2e;
|
||||||
|
color: #34d399;
|
||||||
|
}
|
||||||
|
section.chapter.ch-complete > :not(.ch-toggle) { opacity: 0.4; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 700px) {
|
||||||
|
html { background: #fff; }
|
||||||
|
body {
|
||||||
|
zoom: 1 !important;
|
||||||
|
max-width: 100% !important;
|
||||||
|
padding: 1.25rem 1rem !important;
|
||||||
|
box-shadow: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Cover/Part-Divider: feste mm-Höhen + flex-center fürs Print-A4. Auf Mobile läuft
|
||||||
|
zentrierter Inhalt über die Box hinaus und überlappt das nächste Element.
|
||||||
|
→ als normale Blöcke von oben fließen lassen. */
|
||||||
|
.cover, .part-divider {
|
||||||
|
height: auto !important;
|
||||||
|
min-height: 0 !important;
|
||||||
|
justify-content: flex-start !important;
|
||||||
|
padding: 2.5rem 1.25rem !important;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.cover h1 { font-size: 30pt !important; }
|
||||||
|
.cover h1 .light { font-size: 22pt !important; }
|
||||||
|
.cover .sub { font-size: 12pt !important; max-width: 100% !important; margin-bottom: 1.5rem !important; }
|
||||||
|
.cover .meta-row { flex-wrap: wrap; gap: 0.75rem 1.5rem !important; }
|
||||||
|
.cover-deco { font-size: 60pt !important; right: 1rem !important; }
|
||||||
|
|
||||||
|
.part-divider h1 { font-size: 26pt !important; }
|
||||||
|
.part-divider .part-desc { max-width: 100% !important; }
|
||||||
|
|
||||||
|
.chapter-head h1 { font-size: 22pt !important; }
|
||||||
|
|
||||||
|
/* Print schneidet Code hart ab (overflow: hidden) → auf Mobile umbrechen + scrollbar */
|
||||||
|
pre {
|
||||||
|
white-space: pre-wrap !important;
|
||||||
|
word-break: break-word;
|
||||||
|
overflow-x: auto !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Blocksatz erzeugt riesige Wortlücken in schmalen Spalten */
|
||||||
|
p { text-align: left !important; }
|
||||||
|
|
||||||
|
/* Breite Tabellen horizontal scrollbar statt überlaufend */
|
||||||
|
table { display: block; width: 100%; overflow-x: auto; }
|
||||||
|
}`
|
||||||
|
doc.head?.appendChild(style)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function setupProgress(doc) {
|
||||||
|
const guideId = props.previewGuide?.id
|
||||||
|
const chapters = Array.from(doc.querySelectorAll('section.chapter'))
|
||||||
|
if (!guideId || !chapters.length) return
|
||||||
|
|
||||||
|
let done = new Set()
|
||||||
|
try {
|
||||||
|
const res = await fetchProgress(guideId)
|
||||||
|
done = new Set(res.chapters || [])
|
||||||
|
} catch { /* offline → leer */ }
|
||||||
|
|
||||||
|
chapters.forEach((section, i) => {
|
||||||
|
if (section.querySelector(':scope > .ch-toggle')) return // Guard gegen Doppel-Inject
|
||||||
|
const numEl = section.querySelector('.chapter-num')
|
||||||
|
const key = (numEl?.textContent.match(/\d+/)?.[0]) || String(i + 1)
|
||||||
|
|
||||||
|
const toggle = doc.createElement('div')
|
||||||
|
toggle.className = 'ch-toggle'
|
||||||
|
const apply = (isDone) => {
|
||||||
|
toggle.classList.toggle('is-done', isDone)
|
||||||
|
section.classList.toggle('ch-complete', isDone)
|
||||||
|
toggle.textContent = isDone ? '✓ Erledigt – rückgängig' : 'Kapitel als erledigt markieren'
|
||||||
|
}
|
||||||
|
apply(done.has(key))
|
||||||
|
|
||||||
|
toggle.addEventListener('click', async () => {
|
||||||
|
const newState = !section.classList.contains('ch-complete')
|
||||||
|
apply(newState)
|
||||||
|
try {
|
||||||
|
await setProgress(guideId, key, newState)
|
||||||
|
} catch {
|
||||||
|
apply(!newState) // Rollback bei Fehler
|
||||||
|
}
|
||||||
|
})
|
||||||
|
section.appendChild(toggle)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Zum ersten noch offenen Kapitel springen — aber nur, wenn davor schon was erledigt ist.
|
||||||
|
// Frischer Guide (nichts erledigt) bleibt oben beim Cover.
|
||||||
|
const firstOpen = chapters.find((s) => !s.classList.contains('ch-complete'))
|
||||||
|
if (firstOpen && firstOpen !== chapters[0]) {
|
||||||
|
firstOpen.scrollIntoView({ block: 'start' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Chat ---
|
||||||
|
const chatOpen = ref(false)
|
||||||
|
const messages = ref([])
|
||||||
|
const input = ref('')
|
||||||
|
const loading = ref(false)
|
||||||
|
const messagesEl = ref(null)
|
||||||
|
const inputEl = ref(null)
|
||||||
|
const panelEl = ref(null)
|
||||||
|
|
||||||
|
function openChat() {
|
||||||
|
chatOpen.value = true
|
||||||
|
nextTick(() => inputEl.value?.focus())
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeChat() {
|
||||||
|
chatOpen.value = false
|
||||||
|
messages.value = []
|
||||||
|
input.value = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
// Klicks außerhalb des iframes (Sidebar, Ränder) — mousedown vermeidet die Open/Close-Race beim FAB
|
||||||
|
function onDocMouseDown(e) {
|
||||||
|
if (!chatOpen.value) return
|
||||||
|
if (panelEl.value && panelEl.value.contains(e.target)) return
|
||||||
|
closeChat()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enter öffnet den Chat (wenn zu, nicht in Eingabefeld); ESC schließt ihn
|
||||||
|
function onDocKeyDown(e) {
|
||||||
|
if (e.key === 'Escape' && chatOpen.value) {
|
||||||
|
e.preventDefault()
|
||||||
|
closeChat()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (e.key !== 'Enter' || chatOpen.value || !props.previewGuide) return
|
||||||
|
const tag = document.activeElement?.tagName
|
||||||
|
if (tag === 'INPUT' || tag === 'TEXTAREA') return
|
||||||
|
e.preventDefault()
|
||||||
|
openChat()
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
document.addEventListener('mousedown', onDocMouseDown, true)
|
||||||
|
document.addEventListener('keydown', onDocKeyDown)
|
||||||
|
})
|
||||||
|
onUnmounted(() => {
|
||||||
|
document.removeEventListener('mousedown', onDocMouseDown, true)
|
||||||
|
document.removeEventListener('keydown', onDocKeyDown)
|
||||||
|
})
|
||||||
|
|
||||||
|
function extractContext() {
|
||||||
|
try {
|
||||||
|
const doc = frameEl.value?.contentDocument
|
||||||
|
if (!doc) return { section: '', outline: '' }
|
||||||
|
const headings = Array.from(doc.querySelectorAll('h1, h2, h3'))
|
||||||
|
if (!headings.length) return { section: '', outline: '' }
|
||||||
|
|
||||||
|
const outline = headings.map((h) => h.innerText.trim()).filter(Boolean).join('\n')
|
||||||
|
|
||||||
|
// Aktuelle Überschrift = letzte, die oben im sichtbaren Bereich oder darüber liegt
|
||||||
|
let current = headings[0]
|
||||||
|
for (const h of headings) {
|
||||||
|
if (h.getBoundingClientRect().top <= 100) current = h
|
||||||
|
else break
|
||||||
|
}
|
||||||
|
const level = Number(current.tagName[1])
|
||||||
|
const idx = headings.indexOf(current)
|
||||||
|
let end = null
|
||||||
|
for (let i = idx + 1; i < headings.length; i++) {
|
||||||
|
if (Number(headings[i].tagName[1]) <= level) { end = headings[i]; break }
|
||||||
|
}
|
||||||
|
const range = doc.createRange()
|
||||||
|
range.setStartBefore(current)
|
||||||
|
if (end) range.setEndBefore(end)
|
||||||
|
else range.setEndAfter(doc.body.lastElementChild || doc.body)
|
||||||
|
const section = range.toString().trim().slice(0, 18000)
|
||||||
|
return { section, outline: outline.slice(0, 7000) }
|
||||||
|
} catch {
|
||||||
|
return { section: '', outline: '' }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function scrollToBottom() {
|
||||||
|
await nextTick()
|
||||||
|
if (messagesEl.value) messagesEl.value.scrollTop = messagesEl.value.scrollHeight
|
||||||
|
}
|
||||||
|
|
||||||
|
async function send() {
|
||||||
|
const text = input.value.trim()
|
||||||
|
if (!text || loading.value || !props.previewGuide) return
|
||||||
|
messages.value.push({ role: 'user', content: text })
|
||||||
|
input.value = ''
|
||||||
|
loading.value = true
|
||||||
|
scrollToBottom()
|
||||||
|
try {
|
||||||
|
const { section, outline } = extractContext()
|
||||||
|
const res = await chatGuide(props.previewGuide.id, {
|
||||||
|
section,
|
||||||
|
outline,
|
||||||
|
messages: messages.value,
|
||||||
|
provider: props.provider,
|
||||||
|
})
|
||||||
|
messages.value.push({ role: 'assistant', content: res.reply || '…' })
|
||||||
|
} catch {
|
||||||
|
messages.value.push({ role: 'assistant', content: 'Fehler bei der Anfrage.' })
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
scrollToBottom()
|
||||||
|
nextTick(() => inputEl.value?.focus())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="detail">
|
||||||
|
<div class="preview" :class="{ landscape: isLandscape }" v-if="previewGuide">
|
||||||
|
<iframe ref="frameEl" :src="htmlUrl(previewGuide.id)" class="preview-frame" :class="{ landscape: isLandscape }" title="Guide-Vorschau" @load="onFrameLoad"></iframe>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="empty-preview" v-else>
|
||||||
|
<p>Guide-Format anklicken um zu generieren oder Vorschau zu öffnen.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button v-if="previewGuide && !chatOpen" class="chat-fab" title="Fragen zum Guide" @click="openChat">💬</button>
|
||||||
|
|
||||||
|
<div v-if="previewGuide && chatOpen" ref="panelEl" class="chat-panel">
|
||||||
|
<header class="chat-header">
|
||||||
|
<span>Fragen zum Guide</span>
|
||||||
|
<button class="chat-close" title="Chat beenden" @click="closeChat">×</button>
|
||||||
|
</header>
|
||||||
|
<div ref="messagesEl" class="chat-messages">
|
||||||
|
<p v-if="!messages.length" class="chat-hint">Stell eine Frage zum aktuellen Abschnitt.</p>
|
||||||
|
<template v-for="(m, i) in messages" :key="i">
|
||||||
|
<div v-if="m.role === 'assistant'" class="chat-msg assistant markdown" v-html="renderMarkdown(m.content)"></div>
|
||||||
|
<div v-else class="chat-msg user">{{ m.content }}</div>
|
||||||
|
</template>
|
||||||
|
<div v-if="loading" class="chat-msg assistant chat-typing">Denkt…</div>
|
||||||
|
</div>
|
||||||
|
<div class="chat-input">
|
||||||
|
<textarea
|
||||||
|
ref="inputEl"
|
||||||
|
v-model="input"
|
||||||
|
placeholder="Frage stellen…"
|
||||||
|
@keydown.enter.exact.prevent="send"
|
||||||
|
></textarea>
|
||||||
|
<button :disabled="!input.trim() || loading" @click="send">➤</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.detail {
|
||||||
|
flex: 1;
|
||||||
|
height: 100vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview {
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
background: var(--bg-preview);
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview.landscape {
|
||||||
|
padding: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0.4rem 1rem;
|
||||||
|
background: var(--bg);
|
||||||
|
border-bottom: 1px solid var(--border);
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-muted);
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-link {
|
||||||
|
color: var(--accent);
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-link:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-frame {
|
||||||
|
width: 100%;
|
||||||
|
flex: 1;
|
||||||
|
border: none;
|
||||||
|
background: var(--bg-preview);
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-frame.landscape {
|
||||||
|
max-width: 1180px;
|
||||||
|
box-shadow: 0 1px 8px var(--shadow);
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-preview {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100%;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-fab {
|
||||||
|
position: fixed;
|
||||||
|
right: 1.5rem;
|
||||||
|
bottom: 1.5rem;
|
||||||
|
width: 52px;
|
||||||
|
height: 52px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
font-size: 1.4rem;
|
||||||
|
cursor: pointer;
|
||||||
|
box-shadow: 0 2px 12px var(--shadow);
|
||||||
|
z-index: 20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-fab:hover {
|
||||||
|
background: var(--accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-panel {
|
||||||
|
position: fixed;
|
||||||
|
right: 1.5rem;
|
||||||
|
bottom: 1.5rem;
|
||||||
|
width: 360px;
|
||||||
|
height: 500px;
|
||||||
|
max-height: calc(100vh - 3rem);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background: var(--panel);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 12px;
|
||||||
|
box-shadow: 0 4px 24px var(--shadow);
|
||||||
|
z-index: 20;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 0.6rem 0.9rem;
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-close {
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
color: var(--on-accent);
|
||||||
|
font-size: 1.4rem;
|
||||||
|
line-height: 1;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-messages {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 0.9rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-hint {
|
||||||
|
color: var(--text-faint);
|
||||||
|
font-size: 0.82rem;
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg {
|
||||||
|
max-width: 85%;
|
||||||
|
padding: 7px 11px;
|
||||||
|
border-radius: 12px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
line-height: 1.4;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.user {
|
||||||
|
align-self: flex-end;
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
border-bottom-right-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.assistant {
|
||||||
|
align-self: flex-start;
|
||||||
|
background: var(--panel-soft);
|
||||||
|
color: var(--text);
|
||||||
|
border-bottom-left-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-typing {
|
||||||
|
color: var(--text-faint);
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(p) {
|
||||||
|
margin: 0 0 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(p:last-child) {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(ul),
|
||||||
|
.chat-msg.markdown :deep(ol) {
|
||||||
|
margin: 0.3em 0;
|
||||||
|
padding-left: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(li) {
|
||||||
|
margin: 0.15em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(code) {
|
||||||
|
background: var(--border);
|
||||||
|
padding: 1px 4px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 0.8em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(pre) {
|
||||||
|
background: #1e2330;
|
||||||
|
color: #e6e8ee;
|
||||||
|
padding: 8px 10px;
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow-x: auto;
|
||||||
|
margin: 0.5em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(pre code) {
|
||||||
|
background: none;
|
||||||
|
padding: 0;
|
||||||
|
color: inherit;
|
||||||
|
font-size: 0.78em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(h1),
|
||||||
|
.chat-msg.markdown :deep(h2),
|
||||||
|
.chat-msg.markdown :deep(h3) {
|
||||||
|
font-size: 0.95em;
|
||||||
|
margin: 0.4em 0 0.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(a) {
|
||||||
|
color: var(--accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(table) {
|
||||||
|
border-collapse: collapse;
|
||||||
|
font-size: 0.95em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-msg.markdown :deep(th),
|
||||||
|
.chat-msg.markdown :deep(td) {
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
padding: 2px 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-input {
|
||||||
|
display: flex;
|
||||||
|
gap: 6px;
|
||||||
|
padding: 0.6rem;
|
||||||
|
border-top: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-input textarea {
|
||||||
|
flex: 1;
|
||||||
|
resize: none;
|
||||||
|
height: 38px;
|
||||||
|
padding: 8px 10px;
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
font-family: inherit;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-input textarea:focus {
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-input button {
|
||||||
|
width: 38px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 8px;
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
font-size: 1rem;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chat-input button:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
574
frontend/src/components/TopicSidebar.vue
Normal file
574
frontend/src/components/TopicSidebar.vue
Normal file
@@ -0,0 +1,574 @@
|
|||||||
|
<script setup>
|
||||||
|
import { ref, computed } from 'vue'
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
topics: { type: Array, required: true },
|
||||||
|
projects: { type: Array, default: () => [] },
|
||||||
|
selectedTopic: { type: String, default: null },
|
||||||
|
doneByFormat: { type: Object, default: () => ({}) },
|
||||||
|
latestByFormat: { type: Object, default: () => ({}) },
|
||||||
|
allGuides: { type: Array, default: () => [] },
|
||||||
|
pinned: { type: Boolean, default: true },
|
||||||
|
dark: { type: Boolean, default: false },
|
||||||
|
provider: { type: String, default: 'claude' },
|
||||||
|
providers: { type: Array, default: () => [] },
|
||||||
|
})
|
||||||
|
|
||||||
|
const emit = defineEmits(['select', 'create', 'formatClick', 'deleteTopic', 'deleteProject', 'cancelGuide', 'deleteGuide', 'preview', 'togglePin', 'sidebarLeave', 'toggleDark', 'setProvider'])
|
||||||
|
|
||||||
|
function providerAvailable(id) {
|
||||||
|
const p = props.providers.find((x) => x.id === id)
|
||||||
|
return p ? p.available : true
|
||||||
|
}
|
||||||
|
|
||||||
|
const PROVIDER_LABELS = { claude: 'Claude', minimax: 'MiniMax' }
|
||||||
|
|
||||||
|
const formats = [
|
||||||
|
{ key: 'OnePager', label: 'OnePager' },
|
||||||
|
{ key: 'MiniGuide', label: 'MiniGuide' },
|
||||||
|
{ key: 'Guide', label: 'Guide' },
|
||||||
|
]
|
||||||
|
|
||||||
|
const activeGenerations = computed(() => {
|
||||||
|
return props.allGuides
|
||||||
|
.filter((g) => g.status === 'generating' || g.status === 'queued')
|
||||||
|
.map((g) => `${g.topic} – ${g.format}: ${g.progress || 'Wartend…'}`)
|
||||||
|
})
|
||||||
|
|
||||||
|
function guideStatus(format) {
|
||||||
|
if (props.doneByFormat[format]) return 'done'
|
||||||
|
const latest = props.latestByFormat[format]
|
||||||
|
if (!latest) return 'none'
|
||||||
|
if (latest.status === 'error') return 'none'
|
||||||
|
return latest.status
|
||||||
|
}
|
||||||
|
|
||||||
|
function errorMsg(format) {
|
||||||
|
const latest = props.latestByFormat[format]
|
||||||
|
if (latest?.status === 'error') return latest.error_msg || 'Fehler bei der Generierung'
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleFormatClick(format) {
|
||||||
|
const guide = props.doneByFormat[format]
|
||||||
|
if (guide) {
|
||||||
|
emit('preview', guide)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const activeInput = ref(null)
|
||||||
|
const inputText = ref('')
|
||||||
|
|
||||||
|
function toggleInput(format) {
|
||||||
|
if (activeInput.value === format) {
|
||||||
|
activeInput.value = null
|
||||||
|
inputText.value = ''
|
||||||
|
} else {
|
||||||
|
activeInput.value = format
|
||||||
|
inputText.value = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handlePlay(format) {
|
||||||
|
const text = activeInput.value === format ? inputText.value.trim() : ''
|
||||||
|
emit('formatClick', { format, instructions: text })
|
||||||
|
activeInput.value = null
|
||||||
|
inputText.value = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
function dismissError(format) {
|
||||||
|
const latest = props.latestByFormat[format]
|
||||||
|
if (latest?.status === 'error') {
|
||||||
|
emit('deleteGuide', latest.id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleDelete(format) {
|
||||||
|
const guide = props.latestByFormat[format]
|
||||||
|
if (!guide) return
|
||||||
|
if (guide.status === 'generating' || guide.status === 'queued') {
|
||||||
|
if (!confirm('Generierung abbrechen?')) return
|
||||||
|
emit('cancelGuide', guide.id)
|
||||||
|
} else {
|
||||||
|
if (!confirm('Guide löschen?')) return
|
||||||
|
emit('deleteGuide', guide.id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const newTopic = ref('')
|
||||||
|
|
||||||
|
function submit() {
|
||||||
|
const t = newTopic.value.trim()
|
||||||
|
if (!t) return
|
||||||
|
emit('create', t)
|
||||||
|
newTopic.value = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
function confirmDeleteTopic(topic) {
|
||||||
|
if (!confirm(`Thema "${topic}" und alle zugehörigen Guides löschen?`)) return
|
||||||
|
emit('deleteTopic', topic)
|
||||||
|
}
|
||||||
|
|
||||||
|
function confirmDeleteProject(name) {
|
||||||
|
if (!confirm(`Projekt "${name}" entfernen?\n\nAchtung: Der Quellordner ./projects/${name} wird gelöscht.`)) return
|
||||||
|
emit('deleteProject', name)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<aside class="sidebar" @mouseleave="emit('sidebarLeave')">
|
||||||
|
<div class="new-topic">
|
||||||
|
<button
|
||||||
|
class="pin-btn"
|
||||||
|
:title="pinned ? 'Sidebar ausblenden' : 'Sidebar fixieren'"
|
||||||
|
@click="emit('togglePin')"
|
||||||
|
>{{ pinned ? '⇤' : '⇥' }}</button>
|
||||||
|
<button
|
||||||
|
class="theme-btn"
|
||||||
|
:title="dark ? 'Hellmodus' : 'Dunkelmodus'"
|
||||||
|
@click="emit('toggleDark')"
|
||||||
|
>{{ dark ? '☀' : '🌙' }}</button>
|
||||||
|
<input
|
||||||
|
v-model="newTopic"
|
||||||
|
placeholder="Neues Thema…"
|
||||||
|
@keyup.enter="submit"
|
||||||
|
/>
|
||||||
|
<button @click="submit" :disabled="!newTopic.trim()">+</button>
|
||||||
|
</div>
|
||||||
|
<div class="provider-toggle" v-if="providers.length">
|
||||||
|
<button
|
||||||
|
v-for="p in providers"
|
||||||
|
:key="p.id"
|
||||||
|
:class="{ active: p.id === provider }"
|
||||||
|
:disabled="!p.available"
|
||||||
|
:title="p.available ? '' : 'Nicht konfiguriert (CLI/Key fehlt)'"
|
||||||
|
@click="emit('setProvider', p.id)"
|
||||||
|
>{{ PROVIDER_LABELS[p.id] || p.id }}</button>
|
||||||
|
</div>
|
||||||
|
<div class="format-section" v-if="selectedTopic">
|
||||||
|
<div class="progress-info" v-if="activeGenerations.length">
|
||||||
|
<div v-for="(line, i) in activeGenerations" :key="i">{{ line }}</div>
|
||||||
|
</div>
|
||||||
|
<div v-for="f in formats" :key="f.key">
|
||||||
|
<div :class="['format-row', 'fmt-' + guideStatus(f.key)]">
|
||||||
|
<button class="format-name" @click="handleFormatClick(f.key)">
|
||||||
|
<span class="format-label">{{ f.label }}</span>
|
||||||
|
<span
|
||||||
|
v-if="guideStatus(f.key) !== 'none'"
|
||||||
|
class="format-x"
|
||||||
|
@click.stop="handleDelete(f.key)"
|
||||||
|
:title="guideStatus(f.key) === 'generating' || guideStatus(f.key) === 'queued' ? 'Abbrechen' : 'Löschen'"
|
||||||
|
>×</span>
|
||||||
|
</button>
|
||||||
|
<div class="format-actions">
|
||||||
|
<template v-if="guideStatus(f.key) !== 'generating' && guideStatus(f.key) !== 'queued'">
|
||||||
|
<button class="action-btn play" title="Generieren" @click="handlePlay(f.key)">▶</button>
|
||||||
|
<button
|
||||||
|
class="action-btn pencil"
|
||||||
|
:class="{ active: activeInput === f.key }"
|
||||||
|
title="Anweisungen"
|
||||||
|
@click="toggleInput(f.key)"
|
||||||
|
>✎</button>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-if="errorMsg(f.key)" class="format-error">
|
||||||
|
<span class="format-error-text">{{ errorMsg(f.key) }}</span>
|
||||||
|
<button class="format-error-x" title="Fehler entfernen" @click="dismissError(f.key)">×</button>
|
||||||
|
</div>
|
||||||
|
<div v-if="activeInput === f.key" class="format-input">
|
||||||
|
<input
|
||||||
|
v-model="inputText"
|
||||||
|
placeholder="Anweisungen (optional)…"
|
||||||
|
@keyup.enter="handlePlay(f.key)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ul class="topic-list">
|
||||||
|
<li
|
||||||
|
v-for="t in topics"
|
||||||
|
:key="t"
|
||||||
|
:class="{ active: t === selectedTopic }"
|
||||||
|
@click="emit('select', t)"
|
||||||
|
>
|
||||||
|
<span>{{ t }}</span>
|
||||||
|
<button class="delete-topic" @click.stop="confirmDeleteTopic(t)" title="Löschen">×</button>
|
||||||
|
</li>
|
||||||
|
<template v-if="projects.length">
|
||||||
|
<li class="projects-divider">Projekte</li>
|
||||||
|
<li
|
||||||
|
v-for="p in projects"
|
||||||
|
:key="'project-' + p"
|
||||||
|
:class="{ active: p === selectedTopic, 'project-item': true }"
|
||||||
|
@click="emit('select', p)"
|
||||||
|
>
|
||||||
|
<span>{{ p }}</span>
|
||||||
|
<button class="delete-topic" @click.stop="confirmDeleteProject(p)" title="Projekt entfernen">×</button>
|
||||||
|
</li>
|
||||||
|
</template>
|
||||||
|
</ul>
|
||||||
|
</aside>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.sidebar {
|
||||||
|
width: 300px;
|
||||||
|
min-width: 300px;
|
||||||
|
background: var(--panel);
|
||||||
|
border-right: 1px solid var(--border);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic {
|
||||||
|
display: flex;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 0.75rem;
|
||||||
|
border-bottom: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic input {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
padding: 6px 8px;
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic input:focus {
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic button {
|
||||||
|
padding: 6px 10px;
|
||||||
|
border: none;
|
||||||
|
background: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: 700;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic button:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic .pin-btn {
|
||||||
|
background: var(--bg);
|
||||||
|
color: var(--text-muted);
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
font-weight: 600;
|
||||||
|
padding: 6px 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic .pin-btn:hover {
|
||||||
|
background: var(--accent-soft);
|
||||||
|
color: var(--accent-hover);
|
||||||
|
border-color: var(--accent-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic .theme-btn {
|
||||||
|
background: var(--bg);
|
||||||
|
color: var(--text-muted);
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
font-weight: 600;
|
||||||
|
padding: 6px 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-topic .theme-btn:hover {
|
||||||
|
background: var(--accent-soft);
|
||||||
|
color: var(--accent-hover);
|
||||||
|
border-color: var(--accent-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle {
|
||||||
|
display: flex;
|
||||||
|
gap: 0;
|
||||||
|
padding: 0.5rem 0.75rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle button {
|
||||||
|
flex: 1;
|
||||||
|
padding: 5px 8px;
|
||||||
|
font-size: 0.78rem;
|
||||||
|
font-weight: 600;
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
background: var(--bg);
|
||||||
|
color: var(--text-muted);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle button:first-child {
|
||||||
|
border-radius: 6px 0 0 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle button:last-child {
|
||||||
|
border-radius: 0 6px 6px 0;
|
||||||
|
border-left: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle button.active {
|
||||||
|
background: var(--accent);
|
||||||
|
border-color: var(--accent);
|
||||||
|
color: var(--on-accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.provider-toggle button:disabled {
|
||||||
|
opacity: 0.4;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list {
|
||||||
|
list-style: none;
|
||||||
|
flex: 1;
|
||||||
|
min-height: 0;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 0.5rem 0;
|
||||||
|
border-top: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list li {
|
||||||
|
padding: 0.6rem 1rem;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
color: var(--text);
|
||||||
|
transition: background 0.15s;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list li:hover {
|
||||||
|
background: var(--accent-soft);
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list li.active {
|
||||||
|
background: var(--accent-soft);
|
||||||
|
color: var(--accent-hover);
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-topic {
|
||||||
|
display: none;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--danger);
|
||||||
|
font-size: 1.1rem;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0 2px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list li:hover .delete-topic {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.projects-divider {
|
||||||
|
cursor: default;
|
||||||
|
font-size: 0.7rem;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.05em;
|
||||||
|
color: var(--text-faint);
|
||||||
|
font-weight: 700;
|
||||||
|
padding: 0.6rem 1rem 0.3rem;
|
||||||
|
margin-top: 0.4rem;
|
||||||
|
border-top: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.projects-divider:hover {
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topic-list li.project-item span::before {
|
||||||
|
content: '📁 ';
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Format section */
|
||||||
|
.format-section {
|
||||||
|
flex-shrink: 0;
|
||||||
|
max-height: 60vh;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 0.5rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-info {
|
||||||
|
padding: 0.4rem 0.75rem;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
color: var(--warning);
|
||||||
|
background: var(--warning-soft);
|
||||||
|
margin-bottom: 0.25rem;
|
||||||
|
animation: pulse 1.5s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0.4rem 0.75rem;
|
||||||
|
transition: background 0.15s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-row:hover {
|
||||||
|
background: var(--panel-soft);
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-name {
|
||||||
|
flex: 1;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: default;
|
||||||
|
color: var(--text-faint);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-x {
|
||||||
|
display: none;
|
||||||
|
color: var(--danger);
|
||||||
|
font-size: 1.1rem;
|
||||||
|
line-height: 1;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-name:hover .format-x {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fmt-done .format-name {
|
||||||
|
color: var(--success);
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
background: var(--success-soft);
|
||||||
|
border: 1px solid var(--success-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.fmt-done .format-name:hover {
|
||||||
|
background: var(--success-soft-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.fmt-generating .format-name,
|
||||||
|
.fmt-queued .format-name {
|
||||||
|
color: var(--warning);
|
||||||
|
background: var(--warning-soft);
|
||||||
|
border: 1px solid var(--warning-border);
|
||||||
|
animation: pulse 1.5s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-error {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 2px 0.75rem 6px calc(0.75rem + 8px);
|
||||||
|
font-size: 0.72rem;
|
||||||
|
color: var(--danger);
|
||||||
|
line-height: 1.3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-error-text {
|
||||||
|
flex: 1;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-error-x {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--danger);
|
||||||
|
font-size: 1rem;
|
||||||
|
line-height: 1;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0 2px;
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-error-x:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 2px;
|
||||||
|
margin-left: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
border-radius: 4px;
|
||||||
|
width: 26px;
|
||||||
|
height: 26px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
transition: all 0.15s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn.play {
|
||||||
|
color: var(--success);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn.play:hover {
|
||||||
|
background: var(--success-soft);
|
||||||
|
border-color: var(--success-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn.pencil {
|
||||||
|
color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn.pencil:hover,
|
||||||
|
.action-btn.pencil.active {
|
||||||
|
background: var(--accent-soft);
|
||||||
|
border-color: var(--accent-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-input {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 4px 0.75rem 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-input input {
|
||||||
|
flex: 1;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border: 1px solid var(--border-strong);
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.format-input input:focus {
|
||||||
|
border-color: var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn:disabled {
|
||||||
|
opacity: 0.35;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-btn:disabled:hover {
|
||||||
|
background: none;
|
||||||
|
border-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse {
|
||||||
|
0%, 100% { opacity: 1; }
|
||||||
|
50% { opacity: 0.65; }
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
782
templates/Format/Guide.md
Normal file
782
templates/Format/Guide.md
Normal file
@@ -0,0 +1,782 @@
|
|||||||
|
# Guide Style System — Authoring & Build Specification
|
||||||
|
|
||||||
|
This document is a **complete, self-contained specification** for producing a polished, book-style guide as a single HTML file that renders to a clean A4 PDF. It is **topic-neutral**: use it for programming subjects (PHP, Godot, Blender) and equally for non-technical ones (nutrition, finance, psychology, communication, language learning, …).
|
||||||
|
|
||||||
|
You will normally be given two things: this specification and possibly one reference HTML file built with it. From those alone you must be able to:
|
||||||
|
|
||||||
|
1. Gather the subject knowledge yourself (research as needed).
|
||||||
|
2. Decide a structure (parts → chapters).
|
||||||
|
3. Write a single HTML file that embeds the CSS from this document verbatim.
|
||||||
|
4. Convert that HTML to PDF.
|
||||||
|
|
||||||
|
Follow this spec exactly. The visual identity depends on small details (spacing, weights, the single accent color), so do not improvise the CSS. You **do** have full freedom over content, structure, length, and which optional building blocks you use.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Output contract
|
||||||
|
|
||||||
|
- **One HTML file**, self-contained: a single `<style>` block in `<head>`, no external CSS, no external JS, no web fonts. Fonts are system fonts only (see §3).
|
||||||
|
- The HTML converts to PDF with **WeasyPrint**. The canonical build command is:
|
||||||
|
```
|
||||||
|
weasyprint guide.html guide.pdf
|
||||||
|
```
|
||||||
|
- Page size is **A4**. All page furniture (page numbers, running header, footer label) is produced by CSS `@page` rules — you never write headers/footers into the body.
|
||||||
|
- The body is laid out as: **Cover → Table of Contents → Part 1 divider → its chapters → Part 2 divider → its chapters → …**
|
||||||
|
|
||||||
|
If WeasyPrint is unavailable, the HTML must still be a valid, good-looking standalone document; the layout is built so it degrades gracefully in a browser too.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Mental model of a guide
|
||||||
|
|
||||||
|
A guide is a short book. The structure is always the same three levels:
|
||||||
|
|
||||||
|
- **Part** — a major thematic section (e.g. "Fundamentals", "Advanced", "Pitfalls"). Each part gets a full-page **divider**. A guide typically has **3–8 parts**.
|
||||||
|
- **Chapter** — one focused topic. Chapters flow continuously down the page (they do not force a page break); each is separated from the previous by spacing and its own heading block. **3–10 chapters per part**. Each chapter is numbered sequentially across the whole guide (Chapter 1, 2, 3 … regardless of part).
|
||||||
|
- **Section** (`<h2>`) and **sub-section** (`<h3>`) — structure inside a chapter.
|
||||||
|
|
||||||
|
### Scope — coverage-driven (top tier)
|
||||||
|
|
||||||
|
**10–30 pages, covering all important building blocks of the topic** — everything a practical user needs in real work. Far broader than the MiniGuide tier ("core building blocks only"). How much that is depends on the topic.
|
||||||
|
|
||||||
|
A **topic inventory** is produced beforehand by a research agent and supplied to you. It is binding: every inventory item must appear in the guide as its own section — do not merge, trim, or drop items. Derive parts and chapters from the inventory; part/chapter counts follow from it, there is no default number. Page count is an outcome within the 10–30 range, never a target. State the resulting scope on the cover (parts · chapters).
|
||||||
|
|
||||||
|
### Voice and content rules (apply to every topic)
|
||||||
|
|
||||||
|
- Write in the **reader's language** (German request → German guide). This spec is in English, but that does not constrain the output language.
|
||||||
|
- Open each chapter with a one- or two-sentence **lead** (`.lead`) that frames why the topic matters.
|
||||||
|
- Explain each new term the first time it appears. Assume an intelligent reader who is new to *this* subject.
|
||||||
|
- Prefer **short, concrete examples** over abstract description. For technical topics that means small code snippets; for non-technical topics it means worked examples, sample dialogues, before/after comparisons, small tables, checklists, or step lists.
|
||||||
|
- **Explanations are short and simple**: max. 3 short main clauses per concept, no nested sentences. Plain language over precision-flexing.
|
||||||
|
- Use **callouts** to highlight tips, warnings, side-notes, and deeper digressions — not more than one or two per chapter. Sentences like "Empfehlung: …", "In der Praxis problematisch: …", "Achtung: …" ALWAYS go into the matching callout (tip/warn/note), never into running explanation text.
|
||||||
|
- Keep prose tight. This is a reference people skim and return to, not an essay.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. The CSS — embed this verbatim
|
||||||
|
|
||||||
|
Paste the following into a single `<style>` element in `<head>`. **Do not rename classes, change spacing units, or restructure rules.** The only thing you customize is the small block of CSS variables in `:root` (see §4).
|
||||||
|
|
||||||
|
```css
|
||||||
|
/* ============================================================
|
||||||
|
GUIDE STYLE SYSTEM · Stylesheet (WeasyPrint)
|
||||||
|
============================================================ */
|
||||||
|
|
||||||
|
/* ---------- PAGE ---------- */
|
||||||
|
@page {
|
||||||
|
size: A4;
|
||||||
|
margin: 20mm 18mm 18mm 18mm;
|
||||||
|
@bottom-center {
|
||||||
|
content: counter(page);
|
||||||
|
font-family: -apple-system, "Segoe UI", sans-serif;
|
||||||
|
font-size: 8pt;
|
||||||
|
color: #888;
|
||||||
|
}
|
||||||
|
@bottom-right {
|
||||||
|
content: var(--footer-label); /* set in :root, see §4 */
|
||||||
|
font-family: -apple-system, "Segoe UI", sans-serif;
|
||||||
|
font-size: 8pt;
|
||||||
|
color: #888;
|
||||||
|
}
|
||||||
|
@top-right {
|
||||||
|
content: string(chaptertitle);
|
||||||
|
font-family: -apple-system, "Segoe UI", sans-serif;
|
||||||
|
font-size: 8pt;
|
||||||
|
color: #aaa;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Cover page: no header/footer */
|
||||||
|
@page cover {
|
||||||
|
margin: 0;
|
||||||
|
@bottom-center { content: none; }
|
||||||
|
@bottom-right { content: none; }
|
||||||
|
@top-right { content: none; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Part-divider pages: no running header (the title is already prominent) */
|
||||||
|
@page chapterstart {
|
||||||
|
@top-right { content: none; }
|
||||||
|
}
|
||||||
|
|
||||||
|
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||||||
|
|
||||||
|
:root {
|
||||||
|
/* ===== CUSTOMIZE PER TOPIC — see §4 ===== */
|
||||||
|
--accent: #777BB4; /* main accent */
|
||||||
|
--accent-dark: #4F5B93; /* darker shade for headings */
|
||||||
|
--accent-darker: #2C3E66; /* darkest, for cover gradient + part titles */
|
||||||
|
--footer-label: "Guide"; /* short title shown bottom-right of every page */
|
||||||
|
|
||||||
|
/* ===== USUALLY LEAVE THESE ALONE ===== */
|
||||||
|
--ink: #1a1a1a;
|
||||||
|
--muted: #5a6470;
|
||||||
|
--line: #d8dde3;
|
||||||
|
--bg-soft: #f5f5fb; /* tint this toward your accent if you like */
|
||||||
|
--code-bg: #1e2a3a;
|
||||||
|
--code-fg: #e6e6e6;
|
||||||
|
--plus: #2c8a3e; /* tip / positive */
|
||||||
|
--minus: #c0392b; /* warning / negative */
|
||||||
|
--neutral: #b8860b; /* deep-dive / aside */
|
||||||
|
}
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
font-family: Charter, "Source Serif Pro", Georgia, serif;
|
||||||
|
color: var(--ink);
|
||||||
|
font-size: 10.5pt;
|
||||||
|
line-height: 1.55;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
COVER
|
||||||
|
============================================================ */
|
||||||
|
.cover {
|
||||||
|
page: cover;
|
||||||
|
page-break-after: always;
|
||||||
|
height: 297mm;
|
||||||
|
background: linear-gradient(150deg, var(--accent-darker) 0%, var(--accent-dark) 45%, var(--accent) 100%);
|
||||||
|
color: white;
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 0 22mm;
|
||||||
|
}
|
||||||
|
.cover-logo {
|
||||||
|
width: 30mm; height: 30mm;
|
||||||
|
background: rgba(255,255,255,0.14);
|
||||||
|
border: 1.5pt solid rgba(255,255,255,0.5);
|
||||||
|
border-radius: 7mm;
|
||||||
|
display: flex; align-items: center; justify-content: center;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 20pt; font-weight: 800;
|
||||||
|
margin-bottom: 14mm;
|
||||||
|
letter-spacing: -1pt;
|
||||||
|
}
|
||||||
|
.cover h1 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 42pt;
|
||||||
|
font-weight: 800;
|
||||||
|
line-height: 1.02;
|
||||||
|
letter-spacing: -1.5pt;
|
||||||
|
margin-bottom: 6mm;
|
||||||
|
}
|
||||||
|
.cover h1 .light { font-weight: 300; display:block; font-size: 30pt; opacity: 0.85; }
|
||||||
|
.cover .sub {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 13pt;
|
||||||
|
font-weight: 400;
|
||||||
|
opacity: 0.9;
|
||||||
|
line-height: 1.5;
|
||||||
|
max-width: 130mm;
|
||||||
|
margin-bottom: 18mm;
|
||||||
|
}
|
||||||
|
.cover .meta-row {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 9.5pt;
|
||||||
|
opacity: 0.85;
|
||||||
|
border-top: 1pt solid rgba(255,255,255,0.3);
|
||||||
|
padding-top: 5mm;
|
||||||
|
display: flex;
|
||||||
|
gap: 8mm;
|
||||||
|
}
|
||||||
|
.cover .meta-row b { font-weight: 700; }
|
||||||
|
.cover-deco {
|
||||||
|
position: absolute;
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 120pt;
|
||||||
|
font-weight: 800;
|
||||||
|
opacity: 0.06;
|
||||||
|
bottom: 10mm;
|
||||||
|
right: 12mm;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
TABLE OF CONTENTS
|
||||||
|
============================================================ */
|
||||||
|
.toc {
|
||||||
|
page-break-after: always;
|
||||||
|
}
|
||||||
|
.toc h2 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 22pt;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--accent-dark);
|
||||||
|
margin-bottom: 8mm;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
.toc-part {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 10pt;
|
||||||
|
font-weight: 800;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 1.5pt;
|
||||||
|
color: var(--accent);
|
||||||
|
margin: 6mm 0 2mm 0;
|
||||||
|
padding-bottom: 1mm;
|
||||||
|
border-bottom: 1pt solid var(--line);
|
||||||
|
}
|
||||||
|
.toc-part:first-of-type { margin-top: 0; }
|
||||||
|
.toc-entry {
|
||||||
|
display: flex;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 9.5pt;
|
||||||
|
margin: 1.6mm 0;
|
||||||
|
align-items: baseline;
|
||||||
|
}
|
||||||
|
.toc-num {
|
||||||
|
color: var(--accent-dark);
|
||||||
|
font-weight: 700;
|
||||||
|
width: 11mm;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
.toc-title { color: var(--ink); }
|
||||||
|
.toc-dots {
|
||||||
|
flex: 1;
|
||||||
|
border-bottom: 1pt dotted var(--line);
|
||||||
|
margin: 0 2mm;
|
||||||
|
transform: translateY(-1mm);
|
||||||
|
}
|
||||||
|
.toc-page { color: var(--muted); font-variant-numeric: tabular-nums; }
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
PART DIVIDER
|
||||||
|
============================================================ */
|
||||||
|
.part-divider {
|
||||||
|
page: chapterstart;
|
||||||
|
page-break-before: always;
|
||||||
|
page-break-after: always;
|
||||||
|
height: 257mm;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
background: var(--bg-soft);
|
||||||
|
border-radius: 4mm;
|
||||||
|
padding: 0 20mm;
|
||||||
|
}
|
||||||
|
.part-divider .part-kicker {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 11pt;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 3pt;
|
||||||
|
color: var(--accent);
|
||||||
|
margin-bottom: 4mm;
|
||||||
|
}
|
||||||
|
.part-divider h1 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 34pt;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--accent-darker);
|
||||||
|
letter-spacing: -1pt;
|
||||||
|
line-height: 1.05;
|
||||||
|
margin-bottom: 6mm;
|
||||||
|
}
|
||||||
|
.part-divider .part-desc {
|
||||||
|
font-size: 12pt;
|
||||||
|
color: var(--muted);
|
||||||
|
font-style: italic;
|
||||||
|
max-width: 120mm;
|
||||||
|
}
|
||||||
|
.part-divider .part-chapters {
|
||||||
|
margin-top: 10mm;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 9.5pt;
|
||||||
|
color: var(--accent-dark);
|
||||||
|
}
|
||||||
|
.part-divider .part-chapters span {
|
||||||
|
display: block;
|
||||||
|
margin: 1.5mm 0;
|
||||||
|
padding-left: 5mm;
|
||||||
|
border-left: 2pt solid var(--accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
CHAPTER
|
||||||
|
============================================================ */
|
||||||
|
.chapter {
|
||||||
|
margin-top: 8mm; /* chapters flow continuously; no forced page break */
|
||||||
|
}
|
||||||
|
.chapter-head {
|
||||||
|
string-set: chaptertitle content();
|
||||||
|
margin-bottom: 6mm;
|
||||||
|
padding-bottom: 3mm;
|
||||||
|
border-bottom: 2pt solid var(--ink);
|
||||||
|
}
|
||||||
|
.chapter-num {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 9pt;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 2pt;
|
||||||
|
color: var(--accent);
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 1.5mm;
|
||||||
|
}
|
||||||
|
.chapter-head h1 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 24pt;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--accent-dark);
|
||||||
|
letter-spacing: -0.5pt;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
IN-FLOW HEADINGS
|
||||||
|
============================================================ */
|
||||||
|
h2 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 14pt;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--accent-dark);
|
||||||
|
margin: 6mm 0 2.5mm 0;
|
||||||
|
page-break-after: avoid;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 11pt;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--ink);
|
||||||
|
margin: 4mm 0 1.5mm 0;
|
||||||
|
page-break-after: avoid;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
margin-bottom: 2.5mm;
|
||||||
|
text-align: justify;
|
||||||
|
hyphens: auto;
|
||||||
|
}
|
||||||
|
p b, li b { color: var(--accent-dark); }
|
||||||
|
ul, ol { margin: 1.5mm 0 3mm 6mm; }
|
||||||
|
li { margin-bottom: 1mm; }
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
CODE (optional — technical topics only, see §7)
|
||||||
|
============================================================ */
|
||||||
|
pre {
|
||||||
|
background: var(--code-bg);
|
||||||
|
color: var(--code-fg);
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 8.3pt;
|
||||||
|
line-height: 1.5;
|
||||||
|
padding: 3mm 4mm;
|
||||||
|
border-radius: 2mm;
|
||||||
|
margin: 2.5mm 0 3.5mm 0;
|
||||||
|
white-space: pre;
|
||||||
|
overflow: hidden;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
}
|
||||||
|
.c { color: #6b8aae; font-style: italic; } /* comment / de-emphasized */
|
||||||
|
.k { color: #ff79c6; } /* keyword / control word */
|
||||||
|
.s { color: #f1c40f; } /* string / number / literal value */
|
||||||
|
.f { color: #50fa7b; } /* function / callable name */
|
||||||
|
.t { color: #8be9fd; } /* type / tag / class name */
|
||||||
|
.v { color: #ffb86c; } /* variable / identifier */
|
||||||
|
.a { color: #bd93f9; } /* attribute / annotation / decorator */
|
||||||
|
|
||||||
|
code.inline {
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 9pt;
|
||||||
|
background: var(--bg-soft);
|
||||||
|
padding: 0.3mm 1.5mm;
|
||||||
|
border-radius: 1mm;
|
||||||
|
color: var(--accent-dark);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
CALLOUTS
|
||||||
|
============================================================ */
|
||||||
|
.callout {
|
||||||
|
border-radius: 2mm;
|
||||||
|
padding: 2.5mm 4mm;
|
||||||
|
margin: 3mm 0;
|
||||||
|
font-size: 10pt;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 6mm 1fr;
|
||||||
|
gap: 3mm;
|
||||||
|
}
|
||||||
|
.callout-icon {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-weight: 800;
|
||||||
|
font-size: 14pt;
|
||||||
|
line-height: 1;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.callout-body > b:first-child {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-size: 8pt;
|
||||||
|
letter-spacing: 1pt;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 1mm;
|
||||||
|
}
|
||||||
|
.callout.tip { background: #e8f4ea; border-left: 3pt solid var(--plus); }
|
||||||
|
.callout.tip .callout-icon, .callout.tip .callout-body > b:first-child { color: var(--plus); }
|
||||||
|
.callout.warn { background: #fdecea; border-left: 3pt solid var(--minus); }
|
||||||
|
.callout.warn .callout-icon, .callout.warn .callout-body > b:first-child { color: var(--minus); }
|
||||||
|
.callout.note { background: var(--bg-soft); border-left: 3pt solid var(--accent); }
|
||||||
|
.callout.note .callout-icon, .callout.note .callout-body > b:first-child { color: var(--accent-dark); }
|
||||||
|
.callout.deep { background: #fff8e8; border-left: 3pt solid var(--neutral); }
|
||||||
|
.callout.deep .callout-icon, .callout.deep .callout-body > b:first-child { color: var(--neutral); }
|
||||||
|
|
||||||
|
/* ============================================================
|
||||||
|
TABLES
|
||||||
|
============================================================ */
|
||||||
|
table {
|
||||||
|
border-collapse: collapse;
|
||||||
|
width: 100%;
|
||||||
|
margin: 2.5mm 0 3.5mm 0;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 9pt;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
}
|
||||||
|
th {
|
||||||
|
background: var(--accent-dark);
|
||||||
|
color: white;
|
||||||
|
text-align: left;
|
||||||
|
padding: 1.8mm 3mm;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
td {
|
||||||
|
padding: 1.5mm 3mm;
|
||||||
|
border-bottom: 1pt solid var(--line);
|
||||||
|
vertical-align: top;
|
||||||
|
}
|
||||||
|
tr:nth-child(even) td { background: var(--bg-soft); }
|
||||||
|
td code, th code {
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 8.2pt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* small helpers */
|
||||||
|
.lead {
|
||||||
|
font-size: 11.5pt;
|
||||||
|
color: var(--muted);
|
||||||
|
font-style: italic;
|
||||||
|
margin-bottom: 4mm;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. The only things you customize: `:root` variables
|
||||||
|
|
||||||
|
Change **four** values per guide; leave the rest unless you have a reason. The accent color is the entire brand: pick one that fits the topic, then derive a darker and darkest shade from it (roughly: accent at ~55% lightness, accent-dark ~40%, accent-darker ~28%, all at similar hue).
|
||||||
|
|
||||||
|
| Variable | What it is | Example (PHP) |
|
||||||
|
|---|---|---|
|
||||||
|
| `--accent` | main accent (kickers, links, code-inline, table tint sources) | `#777BB4` |
|
||||||
|
| `--accent-dark` | headings, table header background | `#4F5B93` |
|
||||||
|
| `--accent-darker` | cover gradient end + part titles | `#2C3E66` |
|
||||||
|
| `--footer-label` | short title shown bottom-right every page; **must be a quoted CSS string** | `"PHP Komplett-Guide"` |
|
||||||
|
|
||||||
|
### Recommended accent palettes (free to ignore or replace)
|
||||||
|
|
||||||
|
These are starting points — three shades each (accent / dark / darker):
|
||||||
|
|
||||||
|
- **Programming / PHP** — `#777BB4` / `#4F5B93` / `#2C3E66` (violet)
|
||||||
|
- **Godot / game dev** — `#478CBF` / `#3A6E97` / `#27496B` (blue)
|
||||||
|
- **Blender / 3D / art** — `#E87D0D` / `#C2670A` / `#8A4906` (orange)
|
||||||
|
- **Finance / business** — `#1F7A4D` / `#155C39` / `#0D3D26` (green)
|
||||||
|
- **Nutrition / health** — `#D2603A` / `#A8492B` / `#74331E` (terracotta)
|
||||||
|
- **Psychology / communication** — `#6A5ACD` / `#52459E` / `#372E6B` (indigo)
|
||||||
|
- **Language learning** — `#C0392B` / `#992D22` / `#641E16` (warm red)
|
||||||
|
- **Neutral / mixed topics** — `#3D5A73` / `#2E4557` / `#1E2E3B` (slate)
|
||||||
|
|
||||||
|
When you change the accent, optionally nudge `--bg-soft` toward the same hue (keep it very light, ~96% lightness) so tinted blocks feel cohesive. Everything else (`--ink`, `--muted`, `--line`, callout colors, code colors) stays fixed — those are deliberately topic-independent.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. HTML skeleton
|
||||||
|
|
||||||
|
The body is a flat sequence of sections. There is no wrapper around chapters; the CSS handles page breaks. Order: cover, TOC, then for each part its divider immediately followed by its chapters.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de"> <!-- set to the guide's language -->
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>… Guide title …</title>
|
||||||
|
<style>
|
||||||
|
/* paste the entire stylesheet from §3 here */
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<!-- COVER -->
|
||||||
|
<section class="cover"> … </section>
|
||||||
|
|
||||||
|
<!-- TABLE OF CONTENTS -->
|
||||||
|
<section class="toc"> … </section>
|
||||||
|
|
||||||
|
<!-- PART 1 -->
|
||||||
|
<section class="part-divider"> … </section>
|
||||||
|
<section class="chapter"> … </section>
|
||||||
|
<section class="chapter"> … </section>
|
||||||
|
<!-- … more chapters … -->
|
||||||
|
|
||||||
|
<!-- PART 2 -->
|
||||||
|
<section class="part-divider"> … </section>
|
||||||
|
<section class="chapter"> … </section>
|
||||||
|
<!-- … etc … -->
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
```
|
||||||
|
|
||||||
|
You may generate this HTML however you like — by hand, or with a small script that concatenates strings. A script helps for long guides because it keeps each chapter in its own readable chunk. If you use a script, **the script is throwaway**; the deliverable is the HTML (and the PDF), not the script.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. The building blocks
|
||||||
|
|
||||||
|
Below is the exact markup for every block. Copy these shapes; fill in content.
|
||||||
|
|
||||||
|
### 6.1 Cover
|
||||||
|
|
||||||
|
```html
|
||||||
|
<section class="cover">
|
||||||
|
<div class="cover-logo">LOGO</div>
|
||||||
|
<h1>MAIN TITLE<span class="light">Subtitle line</span></h1>
|
||||||
|
<div class="sub">One or two sentences describing what the guide covers and who it is for.</div>
|
||||||
|
<div class="meta-row">
|
||||||
|
<span><b>N Parts</b> · M Chapters</span>
|
||||||
|
<span><b>Edition / version</b> · Year</span>
|
||||||
|
<span>Focus: …</span>
|
||||||
|
</div>
|
||||||
|
<div class="cover-deco">◆</div>
|
||||||
|
</section>
|
||||||
|
```
|
||||||
|
|
||||||
|
- **`.cover-logo`** — 1–4 characters or a single symbol that evokes the topic. Examples: `php`, `gd` (Godot), a Blender-style `b`, `€` (finance), `Ψ` (psychology), `EN` (English), `🍎` is **not** allowed (no emoji in the logo — keep it crisp). Prefer short letterforms or a geometric glyph (`◆ ● ▲ ■`).
|
||||||
|
- **`.cover h1`** — the big title. The `<span class="light">` is an optional lighter, smaller second line (e.g. `PHP` then `The Complete Guide`). Drop the span for a single-line title.
|
||||||
|
- **`.cover-deco`** — a giant, very faint background glyph in the bottom-right. Pick something topic-flavored: a code fragment (`<?php`, `{ }`, `def`), a symbol (`€`, `∑`, `Ψ`, `♪`), or a single bold letter. Keep it to a few characters; it is decorative and barely visible by design.
|
||||||
|
- **`.meta-row`** — two to four short facts separated into `<span>`s. Use it to state scope (parts/chapters), edition/year, and the focus. This is where you set the reader's expectations about length and depth.
|
||||||
|
|
||||||
|
### 6.2 Table of contents
|
||||||
|
|
||||||
|
List every part, and under it every chapter, numbered sequentially across the whole guide. There are **no real page numbers** (WeasyPrint does not back-fill them here), so the `.toc-page` element is optional — omit it, or use it only if you maintain numbers yourself. The dotted leader still looks right without a trailing number.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<section class="toc">
|
||||||
|
<h2>Contents</h2>
|
||||||
|
|
||||||
|
<div class="toc-part">Part 1 · Fundamentals</div>
|
||||||
|
<div class="toc-entry">
|
||||||
|
<span class="toc-num">1</span>
|
||||||
|
<span class="toc-title">First chapter title</span>
|
||||||
|
<span class="toc-dots"></span>
|
||||||
|
</div>
|
||||||
|
<div class="toc-entry">
|
||||||
|
<span class="toc-num">2</span>
|
||||||
|
<span class="toc-title">Second chapter title</span>
|
||||||
|
<span class="toc-dots"></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="toc-part">Part 2 · Going Deeper</div>
|
||||||
|
<div class="toc-entry">
|
||||||
|
<span class="toc-num">3</span>
|
||||||
|
<span class="toc-title">…</span>
|
||||||
|
<span class="toc-dots"></span>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6.3 Part divider
|
||||||
|
|
||||||
|
One full page that introduces a part. The chapter list mirrors the TOC entries for that part.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<section class="part-divider">
|
||||||
|
<div class="part-kicker">Part 1</div>
|
||||||
|
<h1>Fundamentals</h1>
|
||||||
|
<div class="part-desc">Italic one-liner describing the arc of this part.</div>
|
||||||
|
<div class="part-chapters">
|
||||||
|
<span>1 · First chapter</span>
|
||||||
|
<span>2 · Second chapter</span>
|
||||||
|
<span>3 · Third chapter</span>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6.4 Chapter
|
||||||
|
|
||||||
|
The `.chapter-head` is special: its text is captured into the **running header** at the top-right of every page in that chapter (via `string-set: chaptertitle content()`). Note that `content()` concatenates the text of **all** children, so the chapter-number span and the title run together in the header (e.g. "Chapter 1Title"). For a clean separator, end the `.chapter-num` text with a trailing separator such as `Chapter 1 · ` (trailing space + middot) — or accept the run-on; both are acceptable. Either way keep the chapter `<h1>` reasonably short so it fits on one header line.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<section class="chapter">
|
||||||
|
<div class="chapter-head">
|
||||||
|
<span class="chapter-num">Chapter 1</span>
|
||||||
|
<h1>Chapter title</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="lead">Framing sentence(s) — why this matters.</p>
|
||||||
|
|
||||||
|
<h2>A section</h2>
|
||||||
|
<p>Body text. Use <b>bold</b> for the key term in a sentence. Inline monospace
|
||||||
|
like <code class="inline">term</code> works for any short literal —
|
||||||
|
a command, a key name, a nutrient, a chord, a German case.</p>
|
||||||
|
|
||||||
|
<h3>A sub-section</h3>
|
||||||
|
<p>…</p>
|
||||||
|
|
||||||
|
<!-- tables, callouts, code, lists as needed -->
|
||||||
|
</section>
|
||||||
|
```
|
||||||
|
|
||||||
|
### 6.5 Callouts
|
||||||
|
|
||||||
|
Four flavors. Each has a short uppercase label as the **first `<b>`** inside `.callout-body`, then body text. The icon column holds one glyph.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<div class="callout tip">
|
||||||
|
<div class="callout-icon">✓</div>
|
||||||
|
<div class="callout-body"><b>LABEL</b>Body text giving a practical tip.</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="callout warn">
|
||||||
|
<div class="callout-icon">!</div>
|
||||||
|
<div class="callout-body"><b>LABEL</b>Body text warning about a trap.</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="callout note">
|
||||||
|
<div class="callout-icon">i</div>
|
||||||
|
<div class="callout-body"><b>LABEL</b>Body text for a neutral side note.</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="callout deep">
|
||||||
|
<div class="callout-icon">◆</div>
|
||||||
|
<div class="callout-body"><b>LABEL</b>Body text for an optional deeper dive.</div>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
Meaning of each flavor (topic-independent):
|
||||||
|
|
||||||
|
| Flavor | Color | Use for |
|
||||||
|
|---|---|---|
|
||||||
|
| `tip` | green, ✓ | best practice, a recommendation, a shortcut |
|
||||||
|
| `warn` | red, ! | a common mistake, risk, or thing to avoid |
|
||||||
|
| `note` | accent, i | a neutral aside, clarification, context |
|
||||||
|
| `deep` | gold, ◆ | optional depth: history, edge case, advanced detail |
|
||||||
|
|
||||||
|
Standard icon entities: tip `✓` (✓), warn `!`, note `i`, deep `◆` (◆). You may substitute a more fitting single glyph, but keep it one character.
|
||||||
|
|
||||||
|
### 6.6 Tables
|
||||||
|
|
||||||
|
Plain `<table>` with a header row. The styling (accent header, zebra rows) is automatic. Keep tables to a few columns so they fit A4 width.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<table>
|
||||||
|
<tr><th>Column A</th><th>Column B</th></tr>
|
||||||
|
<tr><td>value</td><td>value</td></tr>
|
||||||
|
<tr><td>value</td><td>value</td></tr>
|
||||||
|
</table>
|
||||||
|
```
|
||||||
|
|
||||||
|
Tables are one of the most useful blocks for **non-technical** topics too: nutrient comparisons, vocabulary lists, pros/cons, decision matrices, dosage/timing, term glossaries.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. Code blocks — optional, technical topics only
|
||||||
|
|
||||||
|
The dark `<pre>` block and the highlight span classes exist for subjects that genuinely involve code or other monospaced literal text (config, shell commands, formulas). **For non-technical guides, do not use `<pre>` blocks at all** — they would look out of place. Use tables, lists, callouts, and worked examples instead. (`code.inline` is fine everywhere for short literals.)
|
||||||
|
|
||||||
|
When you do use code, you **hand-write the highlighting** by wrapping tokens in spans. The classes are intentionally generic so they map onto any language:
|
||||||
|
|
||||||
|
| Class | Generic meaning | Typical use |
|
||||||
|
|---|---|---|
|
||||||
|
| `.c` | de-emphasized | comments |
|
||||||
|
| `.k` | keyword / control word | `if`, `function`, `class`, `return` |
|
||||||
|
| `.s` | literal value | strings, numbers |
|
||||||
|
| `.f` | callable name | function / method names |
|
||||||
|
| `.t` | type / tag / class | type names, HTML tags, class names |
|
||||||
|
| `.v` | identifier | variables |
|
||||||
|
| `.a` | annotation | attributes, decorators, directives |
|
||||||
|
|
||||||
|
Rules for code blocks:
|
||||||
|
|
||||||
|
- Keep snippets **short** (roughly 2–12 lines). They must fit on one page — `page-break-inside: avoid` is set, so an oversized block can overflow. Split long examples into several captioned blocks.
|
||||||
|
- **Escape HTML inside code**: write `<`, `>`, `&`. This is the most common rendering bug.
|
||||||
|
- Inside the dark block, a comment uses `<span class="c">`. Indentation is literal spaces (the block is `white-space: pre`).
|
||||||
|
- You do not need to highlight every token — highlight the ones that aid reading (keywords, strings, names). Plain text inside `<pre>` is fine and renders in the default light color.
|
||||||
|
|
||||||
|
Example (PHP-flavored, but the pattern is language-agnostic):
|
||||||
|
|
||||||
|
```html
|
||||||
|
<pre><span class="k">function</span> <span class="f">greet</span>(<span class="t">string</span> <span class="v">$name</span>): <span class="t">string</span> {
|
||||||
|
<span class="k">return</span> <span class="s">"Hello, $name"</span>; <span class="c">// interpolation</span>
|
||||||
|
}</pre>
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 8. Pitfalls — read before building (these caused real bugs)
|
||||||
|
|
||||||
|
1. **Escape `<`, `>`, `&` inside `<pre>` and `<code>`.** Unescaped angle brackets silently swallow content or break layout. Always `< > &`.
|
||||||
|
2. **Typographic quotes in prose, straight quotes in attributes.** In body text use the target language's real quotation marks (German `„ … "`, English `" … "`). Never let a quote character inside running text collide with HTML attribute quotes. If you generate the HTML from a script, be careful that closing typographic quotes are not accidentally written as escaped ASCII quotes — that corrupts both the string and any nearby `class="…"`. The safe approach: type real `„ … "` / `" … "` glyphs in prose, and reserve `"` strictly for HTML attributes.
|
||||||
|
3. **`--footer-label` must be a quoted CSS string**, e.g. `--footer-label: "Nutrition Guide";`. An unquoted value breaks the `@page` rule.
|
||||||
|
4. **Keep code blocks and callouts short enough to fit one page.** `page-break-inside: avoid` prevents splitting but cannot shrink an oversized block; it will overflow the page. Break large blocks up.
|
||||||
|
5. **Multibyte text is fine** (umlauts, accents, CJK, symbols) — the fonts and UTF-8 charset handle it. But if you ever measure string length in a generator script, count characters, not bytes.
|
||||||
|
6. **Chapter `<h1>` feeds the running header.** Keep it short; a very long chapter title wraps awkwardly in the 8pt header.
|
||||||
|
7. **Don't add web fonts or external assets.** The look depends on the system-font stack already specified. Adding fonts changes metrics and spacing.
|
||||||
|
8. **One accent, three shades.** Don't introduce extra brand colors. Variety comes from the callout colors (green/red/gold), which are fixed and meaningful — not decorative.
|
||||||
|
9. **Tables and code don't split** (`page-break-inside: avoid`). If a table is very long, either let it be its own short section or split it into two.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 9. Build & verify
|
||||||
|
|
||||||
|
1. Write `guide.html` (single file, embedded `<style>`, UTF-8).
|
||||||
|
2. Convert:
|
||||||
|
```
|
||||||
|
weasyprint guide.html guide.pdf
|
||||||
|
```
|
||||||
|
3. **Verify visually.** Render a few pages to images and look at them — do not trust the HTML alone:
|
||||||
|
```
|
||||||
|
pdftoppm -png -r 80 -f 1 -l 1 guide.pdf cover # cover
|
||||||
|
pdftoppm -png -r 80 -f 2 -l 2 guide.pdf toc # contents
|
||||||
|
pdftoppm -png -r 80 -f 5 -l 6 guide.pdf body # a content spread
|
||||||
|
```
|
||||||
|
Check: cover gradient and title; TOC alignment; a part divider; at least one callout; at least one table; (if technical) one code block with highlighting; the running header showing the chapter title; no block overflowing a page edge; quotes and special characters rendering as glyphs, not boxes.
|
||||||
|
4. If something is off, fix the HTML and rebuild. Two or three iterations is normal.
|
||||||
|
5. Deliver **both** `guide.html` and `guide.pdf`.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 10. Quick checklist
|
||||||
|
|
||||||
|
- [ ] Single self-contained HTML, embedded `<style>`, no external assets
|
||||||
|
- [ ] `:root` accent shades + `--footer-label` set for the topic
|
||||||
|
- [ ] Cover: logo glyph, title, subtitle, meta-row, faint deco glyph
|
||||||
|
- [ ] TOC lists every part and every chapter, sequential numbering
|
||||||
|
- [ ] Each part has a divider; each chapter starts with `.chapter-head` + `.lead`
|
||||||
|
- [ ] Callouts used sparingly (≤ ~2 per chapter), correct flavor
|
||||||
|
- [ ] Explanations ≤ 3 short main clauses; recommendations/warnings live in callouts, not prose
|
||||||
|
- [ ] Tables for comparisons/lists; code blocks only if technical, short, escaped
|
||||||
|
- [ ] Prose in the reader's language; terms explained on first use
|
||||||
|
- [ ] Built with WeasyPrint and **visually verified** by rendering pages
|
||||||
|
- [ ] Both HTML and PDF delivered
|
||||||
|
|
||||||
|
## 11. Dark mode (required)
|
||||||
|
|
||||||
|
The app toggles dark mode by setting `data-theme="dark"` on `<html>` inside its preview. Print/PDF always stays light.
|
||||||
|
|
||||||
|
- Define every color through the `:root` variables only (`--ink`, `--muted`, `--line`, `--bg-soft`, accent trio).
|
||||||
|
- Ship an additional override block:
|
||||||
|
|
||||||
|
```css
|
||||||
|
@media screen {
|
||||||
|
html[data-theme="dark"] {
|
||||||
|
--ink: #e6e8ee; --muted: #9aa3b2; --line: #2c3038; --bg-soft: #23262e;
|
||||||
|
/* lift accent shades so headings/links stay readable on dark */
|
||||||
|
}
|
||||||
|
html[data-theme="dark"] body { background: #15171c; }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
- Do NOT use `prefers-color-scheme` — the app controls the attribute.
|
||||||
|
- Do not hardcode callout/infobox backgrounds — or darken them explicitly in the dark block (e.g. to `var(--bg-soft)`); the colored border stays.
|
||||||
|
- Keep dark rules inside `@media screen` only, so WeasyPrint/PDF renders the light theme.
|
||||||
|
- Elements with light text on accent backgrounds (table headers, cover) may keep their light-theme background.
|
||||||
170
templates/Format/MiniGuide.md
Normal file
170
templates/Format/MiniGuide.md
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
```
|
||||||
|
MINI-GUIDE STYLE (HTML/CSS → PDF via WeasyPrint)
|
||||||
|
|
||||||
|
FORMAT
|
||||||
|
- A4 portrait, 3-5 pages
|
||||||
|
- @page { size: A4; margin: 18mm 18mm 16mm 18mm; }
|
||||||
|
- Footer: page number center, guide title right
|
||||||
|
|
||||||
|
COVERAGE (lowest guide tier)
|
||||||
|
- ALL core building blocks of the topic — what you need to get started.
|
||||||
|
Which ones those are depends on the topic.
|
||||||
|
- The topic inventory is produced beforehand by a research agent and
|
||||||
|
supplied. It is binding: every item gets a section.
|
||||||
|
- Sections without chapter structure; code examples very short (2-7 lines)
|
||||||
|
- ~15-25 min reading time, 3-5 pages — this upper limit always applies
|
||||||
|
|
||||||
|
TARGET AUDIENCE — COMPACT BEGINNER INTRO
|
||||||
|
- Real beginners with no programming experience in the topic
|
||||||
|
- Assumes only general understanding ("what is programming")
|
||||||
|
- Terms are explained on first appearance
|
||||||
|
- NO language-character overview for experienced developers
|
||||||
|
- NO advanced features (even if cool and short)
|
||||||
|
|
||||||
|
CONTENT PRINCIPLES
|
||||||
|
- Show only absolute basics
|
||||||
|
- Topics someone can reproduce themselves after 15 min
|
||||||
|
- No tooling complexity (package managers, build systems, compilers)
|
||||||
|
- No language specialties (type systems, decorators, generics)
|
||||||
|
- No OOP if possible (or only the most trivial form)
|
||||||
|
- Depth of explanation before breadth of features
|
||||||
|
- Better 5 concepts thoroughly than 15 superficially
|
||||||
|
|
||||||
|
TYPICAL 5-SECTION STRUCTURE
|
||||||
|
1. Starting the language — installation, first file, first program
|
||||||
|
2. Variables — concept + 2-3 basic types
|
||||||
|
3. Control flow — if/else with a simple example
|
||||||
|
4. Lists + iteration — array + loop
|
||||||
|
5. Functions — declaration + call + return value
|
||||||
|
|
||||||
|
(This order builds on itself and ends with something meaningful.)
|
||||||
|
|
||||||
|
STRUCTURE
|
||||||
|
1. Compact head: logo left (16mm), title + subtitle center, badge + time right
|
||||||
|
2. Gap opener: opening question, italic and framed, low-threshold
|
||||||
|
3. 4-6 H2 sections with explanatory text + code example + optional callout
|
||||||
|
|
||||||
|
ELEMENTS
|
||||||
|
- Body text: justify with hyphenation
|
||||||
|
- Code blocks: dark background, syntax highlighting, very short (2-7 lines)
|
||||||
|
- Inline code: light background, accent color
|
||||||
|
- Tables: only if truly necessary (comparison operators etc.)
|
||||||
|
- Callouts in 3 flavors: tip (green), warn (red), note (accent color)
|
||||||
|
|
||||||
|
TYPOGRAPHY
|
||||||
|
- Body: 10.5pt serif (Charter), line-height 1.55
|
||||||
|
- Head h1: 20pt sans-serif bold
|
||||||
|
- H2 section: 13pt sans-serif bold
|
||||||
|
- Code: 8.5pt monospace, line-height 1.5
|
||||||
|
- Inline code: 9pt monospace
|
||||||
|
- Callout labels: 8pt uppercase, letter-spacing 1pt
|
||||||
|
|
||||||
|
COLORS (max 3 + neutrals)
|
||||||
|
- Accent color: strong, leaning on the topic's official color
|
||||||
|
- Accent dark: darker variant for accents
|
||||||
|
- Background soft: light variant of the accent color
|
||||||
|
- Code background: #1e2a3a
|
||||||
|
- Text: #1a1a1a / muted #5a6470 / line #d8dde3
|
||||||
|
- Callout colors: green/red/accent
|
||||||
|
|
||||||
|
GAP OPENER (REQUIRED)
|
||||||
|
- Italic, framed with an accent-color border
|
||||||
|
- Low-threshold question that the guide answers
|
||||||
|
- Sparks interest with a relevant statistic or practical angle
|
||||||
|
- NO technical terms that have not been explained yet
|
||||||
|
- Examples:
|
||||||
|
- "PHP läuft hinter rund drei Viertel aller Webseiten..."
|
||||||
|
- "JavaScript ist die Sprache des Webs – aber wie schreibt man das eigentliche Code..."
|
||||||
|
- "Python ist die beliebteste Anfänger-Sprache..."
|
||||||
|
|
||||||
|
EXPLANATION DEPTH PER CONCEPT
|
||||||
|
- Name the concept (e.g. "Variable")
|
||||||
|
- Explain in one sentence what it is
|
||||||
|
- Code example with comments
|
||||||
|
- Mention language quirks (e.g. "in PHP beginnen Variablen mit $")
|
||||||
|
- NO reference to concepts that come later
|
||||||
|
|
||||||
|
CALLOUT USAGE
|
||||||
|
- tip (green): practice suggestion at the end, encouraging
|
||||||
|
- warn (red): beginner pitfalls ("= vs ==", "vergessenes Semikolon")
|
||||||
|
- note (accent): background info, explanation of a language quirk
|
||||||
|
|
||||||
|
CALLOUT CSS IMPORTANT
|
||||||
|
- .callout-body > b:first-child with display:block for the label
|
||||||
|
- NOT .callout-body b globally with display:block (destroys inline bold)
|
||||||
|
|
||||||
|
GAP CSS IMPORTANT
|
||||||
|
- .gap > b:first-child with display:block for the "FRAGE ZUM EINSTIEG" label
|
||||||
|
- NOT .gap b globally with display:block (destroys inline bold in the
|
||||||
|
question text)
|
||||||
|
|
||||||
|
TOPIC-SPECIFIC ADJUSTMENTS (choose before generating)
|
||||||
|
- Accent color: the topic's official color
|
||||||
|
- Logo letter(s) or abbreviation
|
||||||
|
- Welcome statistic in the gap opener
|
||||||
|
- Choose 4-6 beginner topics (see standard structure)
|
||||||
|
|
||||||
|
REQUIRED ELEMENTS
|
||||||
|
- 1 gap opener at the start
|
||||||
|
- 5-7 code examples (short, 2-7 lines, beginner-friendly)
|
||||||
|
- At least 1 callout (often: warn for a pitfall, tip for practice at the end)
|
||||||
|
- Inline code for technical terms
|
||||||
|
|
||||||
|
AVOID
|
||||||
|
- TOC or cover (oversized for 15 min)
|
||||||
|
- Introductory filler ("In diesem Mini-Guide lernen wir...")
|
||||||
|
- Claims of completeness (belongs in a larger guide)
|
||||||
|
- Reference tables without explanatory text (belongs in a cheatsheet)
|
||||||
|
- Recall or next-step at the end (a mini-guide ends with content)
|
||||||
|
- Topics that are advanced (even if cool):
|
||||||
|
- Type systems, type hints, generics
|
||||||
|
- OOP features (except the most trivial form)
|
||||||
|
- Tooling (package managers, build, linting)
|
||||||
|
- Language specialties (PHP: strict_types, readonly, Composer, PSR-4)
|
||||||
|
- page break in the middle of a code block or callout
|
||||||
|
(page-break-inside: avoid)
|
||||||
|
- More than 3 font sizes per section
|
||||||
|
- Floats or absolute positioning (breaks in WeasyPrint)
|
||||||
|
- Technical terms without explanation
|
||||||
|
- References to concepts that come later
|
||||||
|
- Edge cases and "but" sentences
|
||||||
|
|
||||||
|
GENERATION WITH FEEDBACK LOOP (max 3 iterations)
|
||||||
|
1. Write the HTML
|
||||||
|
2. weasyprint file.html file.pdf
|
||||||
|
3. PDF to PNGs: convert all pages
|
||||||
|
4. View all pages
|
||||||
|
5. Check:
|
||||||
|
- Head clean (logo does not overlap the title)?
|
||||||
|
- Code blocks not torn across page breaks?
|
||||||
|
- Callouts fully visible?
|
||||||
|
- Inline bolds in callouts/gap correct (not rendered as blocks)?
|
||||||
|
- Footer with page number correct?
|
||||||
|
- Would a real beginner understand this?
|
||||||
|
- Were all technical terms explained on first appearance?
|
||||||
|
6. On problems: fix, repeat from step 2
|
||||||
|
7. Output after max 3 iterations
|
||||||
|
|
||||||
|
INSTALLATION
|
||||||
|
- pip install weasyprint pdf2image
|
||||||
|
- apt install poppler-utils
|
||||||
|
```
|
||||||
|
|
||||||
|
DARK MODE (REQUIRED)
|
||||||
|
- Define all colors exclusively via :root variables
|
||||||
|
(--ink, --muted, --line, --bg-soft, accent variables).
|
||||||
|
- Additionally ship a dark-mode block:
|
||||||
|
@media screen {
|
||||||
|
html[data-theme="dark"] { ...dark variable values... }
|
||||||
|
html[data-theme="dark"] body { background: #15171c; }
|
||||||
|
}
|
||||||
|
- The app enables it via data-theme="dark" on <html>.
|
||||||
|
Do NOT use prefers-color-scheme.
|
||||||
|
- Dark values: dark backgrounds (#15171c / #23262e), light text (#e6e8ee),
|
||||||
|
muted lines (#2c3038). Lift accent colors so headings and links stay
|
||||||
|
readable on a dark background (check contrast). Elements with light text on
|
||||||
|
accent backgrounds (e.g. table headers) may keep their light background
|
||||||
|
color.
|
||||||
|
- Do not hardcode callout/infobox backgrounds — or darken them explicitly in
|
||||||
|
dark mode (e.g. to var(--bg-soft)); the colored border stays.
|
||||||
|
- Only inside @media screen — print/PDF stays light.
|
||||||
115
templates/Format/OnePager.md
Normal file
115
templates/Format/OnePager.md
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
```
|
||||||
|
ONEPAGER STYLE LANDSCAPE (HTML/CSS → PDF via WeasyPrint)
|
||||||
|
|
||||||
|
FORMAT
|
||||||
|
- A4 landscape (297mm × 210mm)
|
||||||
|
- @page { size: A4 landscape; margin: 0; }
|
||||||
|
- Padding: 9mm 11mm 9mm 11mm
|
||||||
|
|
||||||
|
LAYOUT
|
||||||
|
- Grid: hero+stats (auto) / divider / main (1fr, 3 columns) / footer (absolute)
|
||||||
|
- Stats bar integrated into the hero (right), saves vertical space
|
||||||
|
- Main: 3 equal-width columns, gap 4mm
|
||||||
|
- Columns internally: flex-column, gap 4mm
|
||||||
|
- 6 thematic blocks distributed (2 per column)
|
||||||
|
|
||||||
|
STRUCTURE (in this order)
|
||||||
|
1. Hero: logo left (22mm), title + subtitle center, 4 stats right
|
||||||
|
2. Divider: 1.5pt black line
|
||||||
|
3. Main grid: 6 blocks in 3 columns
|
||||||
|
4. Footer: colored box with key message + tag (absolute, bottom)
|
||||||
|
|
||||||
|
DIFFERENCES FROM PORTRAIT
|
||||||
|
- 3 columns instead of 2 (more horizontal space)
|
||||||
|
- 6 blocks instead of 4-5
|
||||||
|
- Stats integrated into the hero instead of their own row
|
||||||
|
- Smaller font sizes (9.5pt body instead of 10pt)
|
||||||
|
- More compact code blocks (7pt instead of 7.5pt)
|
||||||
|
|
||||||
|
BLOCK STRUCTURE
|
||||||
|
- Title: 9.5pt bold uppercase, icon left, accent-color underline 2pt
|
||||||
|
- Content: visual, not plain text bullets
|
||||||
|
- Variants: icon list, code block, tile grid, plus/minus columns, type grid
|
||||||
|
|
||||||
|
COLORS (max 3 + neutrals)
|
||||||
|
- Accent color: lean on the topic's official color
|
||||||
|
- Accent dark: for headings
|
||||||
|
- Accent darker: for the footer
|
||||||
|
- Accent color: contrasting
|
||||||
|
- Background soft: light variant of the accent color
|
||||||
|
- Code background: #1e2a3a
|
||||||
|
- Text: #1a1a1a / muted #5a6470 / line #e5e5e5
|
||||||
|
|
||||||
|
TYPOGRAPHY
|
||||||
|
- Body: 9.5pt, line-height 1.4
|
||||||
|
- Hero h1: 20pt bold
|
||||||
|
- Block title: 9.5pt bold uppercase, letter-spacing 0.5pt
|
||||||
|
- Stats number: 14pt bold, label 6.5pt uppercase
|
||||||
|
- Code: 7pt monospace, dark background
|
||||||
|
- Feature text: 8.5pt
|
||||||
|
- Max 3 font sizes per block
|
||||||
|
|
||||||
|
ICONS
|
||||||
|
- SVG inline, stroke instead of fill
|
||||||
|
- 4mm in block titles (smaller than portrait due to compact layout)
|
||||||
|
- 2.8mm in tile icons
|
||||||
|
- currentColor for automatic adaptation
|
||||||
|
|
||||||
|
TOPIC-SPECIFIC ADJUSTMENTS (choose before generating)
|
||||||
|
- Accent color: the topic's official color
|
||||||
|
- Logo letter(s) or abbreviation
|
||||||
|
- 4 stats: topic-relevant numbers
|
||||||
|
- Block selection: the 6 most important aspects for a first overview
|
||||||
|
|
||||||
|
REQUIRED VISUAL ELEMENTS
|
||||||
|
- At least 1 code block
|
||||||
|
- At least 1 tile grid with icons (ecosystem)
|
||||||
|
- At least 1 plus/minus split (modern vs legacy etc.)
|
||||||
|
- Footer as a colored box (visual anchor)
|
||||||
|
- Stats bar in the hero
|
||||||
|
|
||||||
|
AVOID
|
||||||
|
- Plain bullet lists in every block
|
||||||
|
- More than 6 main blocks (landscape already has more space)
|
||||||
|
- More than 3 font sizes
|
||||||
|
- Marketing filler in the hero
|
||||||
|
- Floats or absolute positioning (except for the footer)
|
||||||
|
- Vertically very long blocks (would unbalance the columns)
|
||||||
|
|
||||||
|
GENERATION WITH FEEDBACK LOOP (max 3 iterations)
|
||||||
|
1. Write the HTML
|
||||||
|
2. weasyprint file.html file.pdf
|
||||||
|
3. PDF to PNG: python -c "from pdf2image import convert_from_path; convert_from_path('file.pdf', dpi=120)[0].save('preview.png')"
|
||||||
|
4. View the preview with the Read tool
|
||||||
|
5. Check:
|
||||||
|
- Stats bar does not overlap the title?
|
||||||
|
- 3 columns balanced (similar height)?
|
||||||
|
- Footer not cut off?
|
||||||
|
- Code block readable?
|
||||||
|
- All icons render?
|
||||||
|
6. On problems: fix, repeat from step 2
|
||||||
|
7. Output after max 3 iterations
|
||||||
|
|
||||||
|
INSTALLATION
|
||||||
|
- pip install weasyprint pdf2image
|
||||||
|
- apt install poppler-utils
|
||||||
|
```
|
||||||
|
|
||||||
|
DARK MODE (REQUIRED)
|
||||||
|
- Define all colors exclusively via :root variables
|
||||||
|
(--ink, --muted, --line, --bg-soft, accent variables).
|
||||||
|
- Additionally ship a dark-mode block:
|
||||||
|
@media screen {
|
||||||
|
html[data-theme="dark"] { ...dark variable values... }
|
||||||
|
html[data-theme="dark"] body { background: #15171c; }
|
||||||
|
}
|
||||||
|
- The app enables it via data-theme="dark" on <html>.
|
||||||
|
Do NOT use prefers-color-scheme.
|
||||||
|
- Dark values: dark backgrounds (#15171c / #23262e), light text (#e6e8ee),
|
||||||
|
muted lines (#2c3038). Lift accent colors so headings and links stay
|
||||||
|
readable on a dark background (check contrast). Elements with light text on
|
||||||
|
accent backgrounds (e.g. table headers) may keep their light background
|
||||||
|
color.
|
||||||
|
- Do not hardcode callout/infobox backgrounds — or darken them explicitly in
|
||||||
|
dark mode (e.g. to var(--bg-soft)); the colored border stays.
|
||||||
|
- Only inside @media screen — print/PDF stays light.
|
||||||
1
templates/Prompt/Bausteine-Quelle-Projekt.md
Normal file
1
templates/Prompt/Bausteine-Quelle-Projekt.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Das Thema ist das Projekt unter {project}. Verschaffe dir mit Bash (ls/find) einen Überblick und lies README, Doku-Ordner und den relevanten Quellcode mit dem Read-Tool. Die Bausteine müssen das echte Projekt widerspiegeln, nichts Erfundenes.
|
||||||
1
templates/Prompt/Bausteine-Quelle-Thema.md
Normal file
1
templates/Prompt/Bausteine-Quelle-Thema.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Recherchiere per Websuche die aktuelle Version und aktuelle Fakten zu "{topic}", damit die Bausteine vollständig und aktuell sind.
|
||||||
13
templates/Prompt/Bausteine.md
Normal file
13
templates/Prompt/Bausteine.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
Ermittle die Bausteine (Konzepte/Funktionen/Features) des Themas "{topic}" für einen Lern-Guide.
|
||||||
|
|
||||||
|
{source}
|
||||||
|
|
||||||
|
Teile ALLE Bausteine des Themas in drei Kategorien:
|
||||||
|
- KERN: ohne diese Bausteine kann man das Thema nicht verstehen oder benutzen
|
||||||
|
- WICHTIG: in der echten Praxis nötig, aber nicht Teil des Einstiegs
|
||||||
|
- REST: Spezialfälle, Randthemen, selten Gebrauchtes
|
||||||
|
|
||||||
|
Schreibe NUR die Markdown-Datei nach: {bausteine_path}
|
||||||
|
|
||||||
|
Aufbau der Datei: drei Abschnitte (## KERN, ## WICHTIG, ## REST), darunter je eine durchnummerierte Liste. Pro Baustein: Titel und 1–2 Sätze, was er umfasst.
|
||||||
|
{extra}
|
||||||
19
templates/Prompt/Chat.md
Normal file
19
templates/Prompt/Chat.md
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
Du bist ein hilfreicher Tutor zum Lern-Guide "{topic}" (Format: {format_name}). Ein Leser stellt dir Fragen, während er den Guide liest.
|
||||||
|
|
||||||
|
GLIEDERUNG DES GUIDES:
|
||||||
|
{outline_block}
|
||||||
|
|
||||||
|
AKTUELLER ABSCHNITT, DEN DER LESER GERADE LIEST:
|
||||||
|
{section_block}
|
||||||
|
|
||||||
|
BISHERIGER CHAT-VERLAUF:
|
||||||
|
{transcript}
|
||||||
|
|
||||||
|
Antworte als Assistent auf die letzte Nutzer-Nachricht.
|
||||||
|
|
||||||
|
WICHTIG – Antwortstil:
|
||||||
|
- KURZ und EINFACH: 1–3 Sätze, klare Sprache.
|
||||||
|
- Keine Einleitung, keine Wiederholung der Frage, kein Markdown-Drumherum.
|
||||||
|
- Beantworte nur die Frage; nutze den Abschnitt und die Gliederung als Kontext.
|
||||||
|
|
||||||
|
Gib NUR die Antwort aus, kein Präfix wie "Assistent:".
|
||||||
1
templates/Prompt/Guide-Fakten-Projekt.md
Normal file
1
templates/Prompt/Guide-Fakten-Projekt.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Die Fakten stammen aus dem Projekt unter {project} — lies bei Bedarf Dateien mit Read/Bash nach.
|
||||||
1
templates/Prompt/Guide-Fakten-Thema.md
Normal file
1
templates/Prompt/Guide-Fakten-Thema.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Prüfe unsichere oder veraltbare Fakten (z. B. Versionsnummern) per Websuche.
|
||||||
18
templates/Prompt/Guide.md
Normal file
18
templates/Prompt/Guide.md
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
Erstelle einen Lern-Guide zum Thema "{topic}" im Format "{format_name}".
|
||||||
|
|
||||||
|
Schreibe NUR die HTML-Datei nach: {html_path}
|
||||||
|
Es wird KEINE PDF erzeugt — führe kein weasyprint aus, auch wenn die Spezifikation es erwähnt.
|
||||||
|
|
||||||
|
BAUSTEINE (von einem Recherche-Agenten erstellt, kategorisiert in KERN/WICHTIG/REST):
|
||||||
|
{bausteine}
|
||||||
|
|
||||||
|
ABDECKUNG für dieses Format: Setze {coverage} um — jeder dieser Bausteine wird eine eigene Sektion. Nicht zusammenfassen, nichts weglassen. Die übrigen Kategorien ignorierst du komplett.
|
||||||
|
|
||||||
|
{facts}
|
||||||
|
|
||||||
|
FORMAT-SPEZIFIKATION:
|
||||||
|
{spec}
|
||||||
|
|
||||||
|
REFERENZ-IMPLEMENTIERUNG (NUR Stil-Vorlage: Bausteine, CSS, Tonalität. Umfang und Struktur kommen aus den BAUSTEINEN, nicht aus der Referenz):
|
||||||
|
{reference}
|
||||||
|
{extra}
|
||||||
2385
templates/Referenz/Guide.md
Normal file
2385
templates/Referenz/Guide.md
Normal file
File diff suppressed because it is too large
Load Diff
395
templates/Referenz/MiniGuide.md
Normal file
395
templates/Referenz/MiniGuide.md
Normal file
@@ -0,0 +1,395 @@
|
|||||||
|
```
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>PHP Mini-Guide</title>
|
||||||
|
<style>
|
||||||
|
@page {
|
||||||
|
size: A4;
|
||||||
|
margin: 18mm 18mm 16mm 18mm;
|
||||||
|
@bottom-center {
|
||||||
|
content: counter(page) " / " counter(pages);
|
||||||
|
font-family: -apple-system, "Segoe UI", sans-serif;
|
||||||
|
font-size: 8pt;
|
||||||
|
color: #888;
|
||||||
|
}
|
||||||
|
@bottom-right {
|
||||||
|
content: "PHP Mini-Guide";
|
||||||
|
font-family: -apple-system, "Segoe UI", sans-serif;
|
||||||
|
font-size: 8pt;
|
||||||
|
color: #888;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--php: #777BB4;
|
||||||
|
--php-dark: #4F5B93;
|
||||||
|
--php-darker: #2C3E66;
|
||||||
|
--ink: #1a1a1a;
|
||||||
|
--muted: #5a6470;
|
||||||
|
--line: #d8dde3;
|
||||||
|
--bg-soft: #f5f5fb;
|
||||||
|
--code-bg: #1e2a3a;
|
||||||
|
--code-fg: #e6e6e6;
|
||||||
|
--plus: #2c8a3e;
|
||||||
|
--minus: #c0392b;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Darkmode — die App setzt data-theme="dark" auf <html>; Druck/PDF bleibt hell */
|
||||||
|
@media screen {
|
||||||
|
html[data-theme="dark"] {
|
||||||
|
--ink: #e6e8ee;
|
||||||
|
--muted: #9aa3b2;
|
||||||
|
--line: #2c3038;
|
||||||
|
--bg-soft: #23262e;
|
||||||
|
}
|
||||||
|
html[data-theme="dark"] body { background: #15171c; }
|
||||||
|
html[data-theme="dark"] .callout { background: var(--bg-soft); }
|
||||||
|
}
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
font-family: Charter, "Source Serif Pro", Georgia, serif;
|
||||||
|
color: var(--ink);
|
||||||
|
font-size: 10.5pt;
|
||||||
|
line-height: 1.55;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== HEAD ===== */
|
||||||
|
.head {
|
||||||
|
display: table;
|
||||||
|
width: 100%;
|
||||||
|
border-bottom: 2pt solid var(--ink);
|
||||||
|
padding-bottom: 4mm;
|
||||||
|
margin-bottom: 5mm;
|
||||||
|
}
|
||||||
|
.head-logo, .head-title, .head-meta {
|
||||||
|
display: table-cell;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.head-logo {
|
||||||
|
width: 20mm;
|
||||||
|
padding-right: 5mm;
|
||||||
|
}
|
||||||
|
.head-logo-box {
|
||||||
|
width: 16mm; height: 16mm;
|
||||||
|
background: linear-gradient(135deg, var(--php) 0%, var(--php-dark) 100%);
|
||||||
|
border-radius: 4mm;
|
||||||
|
display: flex; align-items: center; justify-content: center;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 11pt; font-weight: 800;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
.head-title h1 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 20pt;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--php-dark);
|
||||||
|
letter-spacing: -0.5pt;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
.head-title h1 .accent { color: var(--php); }
|
||||||
|
.head-title .subtitle {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 10pt;
|
||||||
|
color: var(--muted);
|
||||||
|
margin-top: 1mm;
|
||||||
|
}
|
||||||
|
.head-meta {
|
||||||
|
width: 35mm;
|
||||||
|
text-align: right;
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
}
|
||||||
|
.head-meta .badge {
|
||||||
|
display: inline-block;
|
||||||
|
background: var(--ink);
|
||||||
|
color: white;
|
||||||
|
padding: 1.2mm 3mm;
|
||||||
|
border-radius: 1.5mm;
|
||||||
|
font-size: 8.5pt;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
.head-meta .time {
|
||||||
|
display: block;
|
||||||
|
font-size: 7.5pt;
|
||||||
|
color: var(--muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 1pt;
|
||||||
|
margin-top: 1mm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== GAP-OPENER ===== */
|
||||||
|
.gap {
|
||||||
|
background: var(--bg-soft);
|
||||||
|
border-left: 3pt solid var(--php);
|
||||||
|
padding: 3.5mm 5mm;
|
||||||
|
margin: 0 0 6mm 0;
|
||||||
|
font-style: italic;
|
||||||
|
font-size: 10.5pt;
|
||||||
|
}
|
||||||
|
.gap > b:first-child {
|
||||||
|
font-style: normal;
|
||||||
|
color: var(--php-dark);
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-size: 8pt;
|
||||||
|
letter-spacing: 1pt;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 1.5mm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== SECTIONS ===== */
|
||||||
|
h2 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 13pt;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--php-dark);
|
||||||
|
margin: 6mm 0 2.5mm 0;
|
||||||
|
page-break-after: avoid;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-size: 10.5pt;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--ink);
|
||||||
|
margin: 4mm 0 1.5mm 0;
|
||||||
|
page-break-after: avoid;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
margin-bottom: 2.5mm;
|
||||||
|
text-align: justify;
|
||||||
|
hyphens: auto;
|
||||||
|
}
|
||||||
|
p b { color: var(--php-dark); }
|
||||||
|
ul, ol { margin: 1.5mm 0 3mm 6mm; }
|
||||||
|
li { margin-bottom: 1mm; }
|
||||||
|
|
||||||
|
/* ===== CODE ===== */
|
||||||
|
pre {
|
||||||
|
background: var(--code-bg);
|
||||||
|
color: var(--code-fg);
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 8.5pt;
|
||||||
|
line-height: 1.5;
|
||||||
|
padding: 3mm 4mm;
|
||||||
|
border-radius: 2mm;
|
||||||
|
margin: 2.5mm 0 3.5mm 0;
|
||||||
|
white-space: pre;
|
||||||
|
overflow: hidden;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
}
|
||||||
|
.c { color: #6b8aae; font-style: italic; }
|
||||||
|
.k { color: #ff79c6; }
|
||||||
|
.s { color: #f1c40f; }
|
||||||
|
.f { color: #50fa7b; }
|
||||||
|
.t { color: #8be9fd; }
|
||||||
|
.v { color: #ffb86c; }
|
||||||
|
|
||||||
|
code.inline {
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 9pt;
|
||||||
|
background: var(--bg-soft);
|
||||||
|
padding: 0.3mm 1.5mm;
|
||||||
|
border-radius: 1mm;
|
||||||
|
color: var(--php-dark);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== CALLOUT ===== */
|
||||||
|
.callout {
|
||||||
|
border-radius: 2mm;
|
||||||
|
padding: 2.5mm 4mm;
|
||||||
|
margin: 3mm 0;
|
||||||
|
font-size: 10pt;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 6mm 1fr;
|
||||||
|
gap: 3mm;
|
||||||
|
}
|
||||||
|
.callout-icon {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
font-weight: 800;
|
||||||
|
font-size: 14pt;
|
||||||
|
line-height: 1;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.callout-body > b:first-child {
|
||||||
|
font-family: -apple-system, sans-serif;
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-size: 8pt;
|
||||||
|
letter-spacing: 1pt;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 1mm;
|
||||||
|
}
|
||||||
|
.callout.tip { background: #e8f4ea; border-left: 3pt solid var(--plus); }
|
||||||
|
.callout.tip .callout-icon, .callout.tip .callout-body > b:first-child { color: var(--plus); }
|
||||||
|
.callout.warn { background: #fdecea; border-left: 3pt solid var(--minus); }
|
||||||
|
.callout.warn .callout-icon, .callout.warn .callout-body > b:first-child { color: var(--minus); }
|
||||||
|
.callout.note { background: var(--bg-soft); border-left: 3pt solid var(--php); }
|
||||||
|
.callout.note .callout-icon, .callout.note .callout-body > b:first-child { color: var(--php-dark); }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<!-- ===== HEAD ===== -->
|
||||||
|
<header class="head">
|
||||||
|
<div class="head-logo"><div class="head-logo-box">php</div></div>
|
||||||
|
<div class="head-title">
|
||||||
|
<h1><span class="accent">PHP</span> in 15 Minuten</h1>
|
||||||
|
<div class="subtitle">Dein erstes PHP-Programm – Schritt für Schritt</div>
|
||||||
|
</div>
|
||||||
|
<div class="head-meta">
|
||||||
|
<span class="badge">Mini-Guide</span>
|
||||||
|
<span class="time">15 Min · von Null</span>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<!-- ===== GAP-OPENER ===== -->
|
||||||
|
<div class="gap">
|
||||||
|
<b>Frage zum Einstieg</b>
|
||||||
|
PHP läuft hinter rund drei Viertel aller Webseiten – inklusive WordPress, Wikipedia und Facebook. Aber wie sieht PHP-Code überhaupt aus, und wie startet man? In 15 Minuten kannst du dein erstes Programm schreiben.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ===== SEKTION 1 ===== -->
|
||||||
|
<h2>PHP starten</h2>
|
||||||
|
|
||||||
|
<p>PHP ist eine Programmiersprache, die auf deinem Computer oder einem Webserver läuft. Im Gegensatz zu HTML oder CSS wird PHP nicht im Browser angezeigt – es <i>erzeugt</i> Ausgaben (zum Beispiel HTML), die dann ausgeliefert werden.</p>
|
||||||
|
|
||||||
|
<p>Um anzufangen, brauchst du PHP auf deinem Computer. Auf Mac: <code class="inline">brew install php</code>. Auf Ubuntu: <code class="inline">apt install php8.4-cli</code>. Auf Windows: am einfachsten WSL2 mit Ubuntu darin nutzen.</p>
|
||||||
|
|
||||||
|
<p>PHP-Code lebt in Dateien mit der Endung <code class="inline">.php</code>. Lege eine Datei <code class="inline">hallo.php</code> an mit diesem Inhalt:</p>
|
||||||
|
|
||||||
|
<pre><span class="t"><?php</span>
|
||||||
|
|
||||||
|
<span class="k">echo</span> <span class="s">"Hallo Welt!"</span>;</pre>
|
||||||
|
|
||||||
|
<p>Die erste Zeile <code class="inline"><?php</code> sagt PHP: "ab hier kommt mein Code". Das Wort <code class="inline">echo</code> heißt: "gib das aus, was danach kommt". Strings (also Text) stehen in Anführungszeichen. Jede Anweisung endet mit einem Semikolon.</p>
|
||||||
|
|
||||||
|
<p>Im Terminal ausführen mit:</p>
|
||||||
|
|
||||||
|
<pre>php hallo.php</pre>
|
||||||
|
|
||||||
|
<p>Du siehst "Hallo Welt!" – dein erstes PHP-Programm läuft.</p>
|
||||||
|
|
||||||
|
<!-- ===== SEKTION 2 ===== -->
|
||||||
|
<h2>Variablen</h2>
|
||||||
|
|
||||||
|
<p>Eine <b>Variable</b> ist ein benannter Platz, an dem du einen Wert speicherst. In PHP beginnen Variablen immer mit einem Dollar-Zeichen <code class="inline">$</code>. Das macht sie im Code sofort erkennbar:</p>
|
||||||
|
|
||||||
|
<pre><span class="t"><?php</span>
|
||||||
|
|
||||||
|
<span class="v">$name</span> = <span class="s">"Marek"</span>;
|
||||||
|
<span class="v">$alter</span> = <span class="s">34</span>;
|
||||||
|
<span class="v">$istAktiv</span> = <span class="k">true</span>;
|
||||||
|
|
||||||
|
<span class="k">echo</span> <span class="v">$name</span>;</pre>
|
||||||
|
|
||||||
|
<p>Variablen können verschiedene Arten von Werten enthalten. Die drei wichtigsten:</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><b>Strings</b> – Text in Anführungszeichen, z.B. <code class="inline">"Hallo"</code></li>
|
||||||
|
<li><b>Zahlen</b> – ganze Zahlen (<code class="inline">42</code>) oder Kommazahlen (<code class="inline">3.14</code>)</li>
|
||||||
|
<li><b>Wahrheitswerte</b> – <code class="inline">true</code> (wahr) oder <code class="inline">false</code> (falsch)</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p>Mit dem Punkt verbindest du Strings:</p>
|
||||||
|
|
||||||
|
<pre><span class="v">$gruss</span> = <span class="s">"Hallo, "</span> . <span class="v">$name</span> . <span class="s">"!"</span>;
|
||||||
|
<span class="k">echo</span> <span class="v">$gruss</span>; <span class="c">// Hallo, Marek!</span></pre>
|
||||||
|
|
||||||
|
<!-- ===== SEKTION 3 ===== -->
|
||||||
|
<h2>Bedingungen</h2>
|
||||||
|
|
||||||
|
<p>Programme müssen oft entscheiden: "wenn X zutrifft, mach Y, sonst Z". Dafür gibt es <code class="inline">if</code> und <code class="inline">else</code>:</p>
|
||||||
|
|
||||||
|
<pre><span class="v">$alter</span> = <span class="s">17</span>;
|
||||||
|
|
||||||
|
<span class="k">if</span> (<span class="v">$alter</span> >= <span class="s">18</span>) {
|
||||||
|
<span class="k">echo</span> <span class="s">"Du bist erwachsen."</span>;
|
||||||
|
} <span class="k">else</span> {
|
||||||
|
<span class="k">echo</span> <span class="s">"Du bist noch minderjährig."</span>;
|
||||||
|
}</pre>
|
||||||
|
|
||||||
|
<p>Die Klammer hinter <code class="inline">if</code> enthält die Bedingung. Die geschweiften Klammern <code class="inline">{ }</code> umschließen den Code, der ausgeführt wird, wenn die Bedingung wahr ist. <code class="inline">else</code> ist der Block, wenn sie falsch ist.</p>
|
||||||
|
|
||||||
|
<p>Wichtige Vergleichs-Operatoren:</p>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li><code class="inline">==</code> gleich</li>
|
||||||
|
<li><code class="inline">!=</code> ungleich</li>
|
||||||
|
<li><code class="inline"><</code>, <code class="inline">></code> kleiner, größer</li>
|
||||||
|
<li><code class="inline"><=</code>, <code class="inline">>=</code> kleiner-gleich, größer-gleich</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div class="callout note">
|
||||||
|
<div class="callout-icon">i</div>
|
||||||
|
<div class="callout-body">
|
||||||
|
<b>Ein Gleichheitszeichen reicht nicht</b>
|
||||||
|
Zum Zuweisen nutzt du <code class="inline">=</code> (ein Gleichheitszeichen). Zum Vergleichen brauchst du <code class="inline">==</code> (zwei). Das ist eine häufige Verwechslung bei Anfängern.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ===== SEKTION 4 ===== -->
|
||||||
|
<h2>Listen und Schleifen</h2>
|
||||||
|
|
||||||
|
<p>Mehrere Werte fasst du in einer <b>Liste</b> zusammen. In PHP heißen Listen <code class="inline">array</code>:</p>
|
||||||
|
|
||||||
|
<pre><span class="v">$obst</span> = [<span class="s">"Apfel"</span>, <span class="s">"Birne"</span>, <span class="s">"Kirsche"</span>];</pre>
|
||||||
|
|
||||||
|
<p>Über jede Liste kannst du mit <code class="inline">foreach</code> Schritt für Schritt gehen:</p>
|
||||||
|
|
||||||
|
<pre><span class="k">foreach</span> (<span class="v">$obst</span> <span class="k">as</span> <span class="v">$frucht</span>) {
|
||||||
|
<span class="k">echo</span> <span class="v">$frucht</span> . <span class="s">"\n"</span>;
|
||||||
|
}</pre>
|
||||||
|
|
||||||
|
<p>Das gibt "Apfel", "Birne", "Kirsche" untereinander aus. Das <code class="inline">\n</code> ist ein Zeilenumbruch. Die Variable <code class="inline">$frucht</code> bekommt bei jedem Durchlauf den nächsten Wert aus der Liste.</p>
|
||||||
|
|
||||||
|
<p>Listen müssen nicht aus Strings bestehen. Zahlen gehen genauso:</p>
|
||||||
|
|
||||||
|
<pre><span class="v">$zahlen</span> = [<span class="s">10</span>, <span class="s">20</span>, <span class="s">30</span>];
|
||||||
|
<span class="v">$summe</span> = <span class="s">0</span>;
|
||||||
|
|
||||||
|
<span class="k">foreach</span> (<span class="v">$zahlen</span> <span class="k">as</span> <span class="v">$zahl</span>) {
|
||||||
|
<span class="v">$summe</span> = <span class="v">$summe</span> + <span class="v">$zahl</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
<span class="k">echo</span> <span class="v">$summe</span>; <span class="c">// 60</span></pre>
|
||||||
|
|
||||||
|
<!-- ===== SEKTION 5 ===== -->
|
||||||
|
<h2>Funktionen</h2>
|
||||||
|
|
||||||
|
<p>Wenn du denselben Code mehrfach brauchst, packst du ihn in eine <b>Funktion</b>. Du gibst der Funktion einen Namen und kannst sie dann beliebig oft aufrufen:</p>
|
||||||
|
|
||||||
|
<pre><span class="k">function</span> <span class="f">begruessen</span>(<span class="v">$name</span>) {
|
||||||
|
<span class="k">echo</span> <span class="s">"Hallo, "</span> . <span class="v">$name</span> . <span class="s">"!\n"</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
<span class="f">begruessen</span>(<span class="s">"Marek"</span>);
|
||||||
|
<span class="f">begruessen</span>(<span class="s">"Anna"</span>);
|
||||||
|
<span class="f">begruessen</span>(<span class="s">"Tom"</span>);</pre>
|
||||||
|
|
||||||
|
<p>Die Funktion <code class="inline">begruessen</code> nimmt einen <b>Parameter</b> entgegen (<code class="inline">$name</code>). Beim Aufruf übergibst du den konkreten Wert in den Klammern.</p>
|
||||||
|
|
||||||
|
<p>Funktionen können auch Werte <b>zurückgeben</b>. Dafür gibt es <code class="inline">return</code>:</p>
|
||||||
|
|
||||||
|
<pre><span class="k">function</span> <span class="f">addiere</span>(<span class="v">$a</span>, <span class="v">$b</span>) {
|
||||||
|
<span class="k">return</span> <span class="v">$a</span> + <span class="v">$b</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
<span class="v">$ergebnis</span> = <span class="f">addiere</span>(<span class="s">3</span>, <span class="s">5</span>);
|
||||||
|
<span class="k">echo</span> <span class="v">$ergebnis</span>; <span class="c">// 8</span></pre>
|
||||||
|
|
||||||
|
<p>Die Funktion macht ihre Berechnung und liefert das Ergebnis zurück. Du fängst es in einer Variable auf und kannst damit weiterarbeiten.</p>
|
||||||
|
|
||||||
|
<div class="callout tip">
|
||||||
|
<div class="callout-icon">✓</div>
|
||||||
|
<div class="callout-body">
|
||||||
|
<b>Übung macht den Meister</b>
|
||||||
|
Schreibe jetzt selbst ein kleines PHP-Programm. Zum Beispiel: eine Liste deiner Lieblings-Filme, die mit <code class="inline">foreach</code> ausgegeben werden. Oder eine Funktion, die das Doppelte einer Zahl zurückgibt. Praktisch ausprobieren ist der schnellste Weg, PHP zu lernen.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
```
|
||||||
538
templates/Referenz/OnePager.md
Normal file
538
templates/Referenz/OnePager.md
Normal file
@@ -0,0 +1,538 @@
|
|||||||
|
```
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>PHP OnePager</title>
|
||||||
|
<style>
|
||||||
|
@page { size: A4 landscape; margin: 0; }
|
||||||
|
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--php: #777BB4;
|
||||||
|
--php-dark: #4F5B93;
|
||||||
|
--php-darker: #2C3E66;
|
||||||
|
--accent: #F1C40F;
|
||||||
|
--ink: #1a1a1a;
|
||||||
|
--muted: #5a6470;
|
||||||
|
--line: #e5e5e5;
|
||||||
|
--bg-soft: #f5f5fb;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Darkmode — die App setzt data-theme="dark" auf <html>; Druck/PDF bleibt hell */
|
||||||
|
@media screen {
|
||||||
|
html[data-theme="dark"] {
|
||||||
|
--ink: #e6e8ee;
|
||||||
|
--muted: #9aa3b2;
|
||||||
|
--line: #2c3038;
|
||||||
|
--bg-soft: #23262e;
|
||||||
|
}
|
||||||
|
html[data-theme="dark"] body { background: #15171c; }
|
||||||
|
html[data-theme="dark"] .callout { background: var(--bg-soft); }
|
||||||
|
}
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
font-family: -apple-system, "Segoe UI", Helvetica, Arial, sans-serif;
|
||||||
|
color: var(--ink);
|
||||||
|
font-size: 9.5pt;
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page {
|
||||||
|
width: 297mm;
|
||||||
|
height: 210mm;
|
||||||
|
padding: 9mm 11mm 9mm 11mm;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== HERO ===== */
|
||||||
|
.hero {
|
||||||
|
display: table;
|
||||||
|
width: 100%;
|
||||||
|
margin-bottom: 4mm;
|
||||||
|
}
|
||||||
|
.hero-logo, .hero-text, .hero-stats {
|
||||||
|
display: table-cell;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.hero-logo { width: 28mm; padding-right: 5mm; }
|
||||||
|
.hero-logo-box {
|
||||||
|
width: 22mm; height: 22mm;
|
||||||
|
background: linear-gradient(135deg, var(--php) 0%, var(--php-dark) 100%);
|
||||||
|
border-radius: 5mm;
|
||||||
|
display: flex; align-items: center; justify-content: center;
|
||||||
|
font-size: 18pt; font-weight: 800;
|
||||||
|
color: white;
|
||||||
|
letter-spacing: -1pt;
|
||||||
|
}
|
||||||
|
.hero h1 {
|
||||||
|
font-size: 20pt;
|
||||||
|
font-weight: 800;
|
||||||
|
letter-spacing: -0.5pt;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
.hero h1 .accent { color: var(--php); }
|
||||||
|
.hero p {
|
||||||
|
font-size: 9.5pt;
|
||||||
|
color: var(--muted);
|
||||||
|
margin-top: 1mm;
|
||||||
|
}
|
||||||
|
.hero-stats {
|
||||||
|
width: 110mm;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
.stats-row {
|
||||||
|
display: inline-table;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
.stat-cell {
|
||||||
|
display: table-cell;
|
||||||
|
padding: 0 5mm;
|
||||||
|
text-align: center;
|
||||||
|
border-right: 0.5pt solid var(--line);
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.stat-cell:last-child { border-right: none; padding-right: 0; }
|
||||||
|
.stat-cell:first-child { padding-left: 0; }
|
||||||
|
.stat-num {
|
||||||
|
font-size: 14pt;
|
||||||
|
font-weight: 800;
|
||||||
|
color: var(--php);
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
.stat-label {
|
||||||
|
font-size: 6.5pt;
|
||||||
|
color: var(--muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5pt;
|
||||||
|
margin-top: 1mm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== Divider ===== */
|
||||||
|
.divider {
|
||||||
|
height: 1.5pt;
|
||||||
|
background: var(--ink);
|
||||||
|
margin-bottom: 4mm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== GRID 3-Spalten ===== */
|
||||||
|
.grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr 1fr;
|
||||||
|
gap: 4mm;
|
||||||
|
margin-bottom: 4mm;
|
||||||
|
}
|
||||||
|
.col { display: flex; flex-direction: column; gap: 4mm; }
|
||||||
|
|
||||||
|
.block { display: flex; flex-direction: column; gap: 1.5mm; }
|
||||||
|
.block h2 {
|
||||||
|
font-size: 9.5pt;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5pt;
|
||||||
|
color: var(--php);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 2mm;
|
||||||
|
border-bottom: 2pt solid var(--php);
|
||||||
|
padding-bottom: 1.2mm;
|
||||||
|
}
|
||||||
|
.block h2 .ico { width: 4mm; height: 4mm; flex-shrink: 0; }
|
||||||
|
|
||||||
|
/* ===== FEATURE LIST ===== */
|
||||||
|
.feature-list { display: flex; flex-direction: column; gap: 1.5mm; }
|
||||||
|
.feature-row {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 4.5mm 1fr;
|
||||||
|
gap: 1.8mm;
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
|
.feature-row .dot {
|
||||||
|
width: 4mm; height: 4mm; margin-top: 0.2mm;
|
||||||
|
background: var(--php);
|
||||||
|
border-radius: 1mm;
|
||||||
|
display: flex; align-items: center; justify-content: center;
|
||||||
|
font-size: 6pt; font-weight: 800; color: white;
|
||||||
|
}
|
||||||
|
.feature-row .text { font-size: 8.5pt; line-height: 1.35; }
|
||||||
|
.feature-row .text b { color: var(--php-dark); }
|
||||||
|
.feature-row code {
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 7.5pt;
|
||||||
|
background: var(--bg-soft);
|
||||||
|
padding: 0 1mm;
|
||||||
|
border-radius: 1mm;
|
||||||
|
color: var(--php-dark);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== CODE ===== */
|
||||||
|
.code {
|
||||||
|
background: #1e2a3a;
|
||||||
|
color: #e6e6e6;
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
font-size: 7pt;
|
||||||
|
padding: 2.5mm 3mm;
|
||||||
|
border-radius: 1.5mm;
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
.code .c { color: #6b8aae; font-style: italic; }
|
||||||
|
.code .k { color: #ff79c6; }
|
||||||
|
.code .s { color: #f1c40f; }
|
||||||
|
.code .f { color: #50fa7b; }
|
||||||
|
.code .t { color: #8be9fd; }
|
||||||
|
.code .v { color: #ffb86c; }
|
||||||
|
|
||||||
|
/* ===== ECOSYSTEM ===== */
|
||||||
|
.eco-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 1.5mm;
|
||||||
|
}
|
||||||
|
.eco-item {
|
||||||
|
background: var(--bg-soft);
|
||||||
|
border-left: 1.5pt solid var(--php);
|
||||||
|
border-radius: 1mm;
|
||||||
|
padding: 1.5mm 2mm;
|
||||||
|
display: flex; align-items: center; gap: 2mm;
|
||||||
|
}
|
||||||
|
.eco-ico {
|
||||||
|
width: 5mm; height: 5mm; flex-shrink: 0;
|
||||||
|
background: var(--php-dark);
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex; align-items: center; justify-content: center;
|
||||||
|
}
|
||||||
|
.eco-text { display: flex; flex-direction: column; gap: 0.2mm; }
|
||||||
|
.eco-name { font-weight: 700; font-size: 8pt; color: var(--php-dark); }
|
||||||
|
.eco-desc { font-size: 6.5pt; color: var(--muted); }
|
||||||
|
|
||||||
|
/* ===== TYPE TABLE ===== */
|
||||||
|
.types {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 1.5mm;
|
||||||
|
}
|
||||||
|
.type {
|
||||||
|
background: var(--bg-soft);
|
||||||
|
padding: 1.3mm 2mm;
|
||||||
|
border-radius: 1mm;
|
||||||
|
font-size: 7.5pt;
|
||||||
|
}
|
||||||
|
.type b {
|
||||||
|
font-family: "SF Mono", Consolas, monospace;
|
||||||
|
color: var(--php-dark);
|
||||||
|
display: block;
|
||||||
|
font-size: 8pt;
|
||||||
|
}
|
||||||
|
.type span { color: var(--muted); font-size: 6.8pt; }
|
||||||
|
|
||||||
|
/* ===== SPLIT (composer/comparison) ===== */
|
||||||
|
.split {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 2mm;
|
||||||
|
}
|
||||||
|
.split-col h4 {
|
||||||
|
font-size: 7.5pt;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.4pt;
|
||||||
|
margin-bottom: 1mm;
|
||||||
|
}
|
||||||
|
.split-col.modern h4 { color: var(--php); }
|
||||||
|
.split-col.legacy h4 { color: var(--muted); }
|
||||||
|
.split-col ul { list-style: none; }
|
||||||
|
.split-col li {
|
||||||
|
font-size: 7.5pt;
|
||||||
|
padding-left: 3mm;
|
||||||
|
position: relative;
|
||||||
|
margin-bottom: 0.8mm;
|
||||||
|
line-height: 1.3;
|
||||||
|
}
|
||||||
|
.split-col.modern li::before {
|
||||||
|
content: "+";
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
color: var(--php);
|
||||||
|
font-weight: 800;
|
||||||
|
}
|
||||||
|
.split-col.legacy li::before {
|
||||||
|
content: "·";
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
color: var(--muted);
|
||||||
|
font-weight: 800;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== FOOTER ===== */
|
||||||
|
.footer {
|
||||||
|
position: absolute;
|
||||||
|
left: 11mm;
|
||||||
|
right: 11mm;
|
||||||
|
bottom: 9mm;
|
||||||
|
background: var(--php-darker);
|
||||||
|
color: white;
|
||||||
|
padding: 3mm 5mm;
|
||||||
|
border-radius: 1.5mm;
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr auto;
|
||||||
|
gap: 4mm;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.footer-text { font-size: 8.5pt; line-height: 1.45; }
|
||||||
|
.footer-text b { color: var(--accent); }
|
||||||
|
.footer-tag {
|
||||||
|
background: var(--php);
|
||||||
|
color: white;
|
||||||
|
padding: 1.5mm 3mm;
|
||||||
|
border-radius: 1.5mm;
|
||||||
|
font-weight: 800;
|
||||||
|
font-size: 8.5pt;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="page">
|
||||||
|
|
||||||
|
<!-- HERO -->
|
||||||
|
<header class="hero">
|
||||||
|
<div class="hero-logo"><div class="hero-logo-box">php</div></div>
|
||||||
|
<div class="hero-text">
|
||||||
|
<h1><span class="accent">PHP</span> – Server-Sprache des Web</h1>
|
||||||
|
<p>Dynamische Webseiten · seit 1995 · 75% aller Websites · objektorientiert & funktional</p>
|
||||||
|
</div>
|
||||||
|
<div class="hero-stats">
|
||||||
|
<div class="stats-row">
|
||||||
|
<div class="stat-cell">
|
||||||
|
<div class="stat-num">8.4</div>
|
||||||
|
<div class="stat-label">Aktuelle Version</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-cell">
|
||||||
|
<div class="stat-num">1995</div>
|
||||||
|
<div class="stat-label">Erstes Release</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-cell">
|
||||||
|
<div class="stat-num">75%</div>
|
||||||
|
<div class="stat-label">Aller Websites</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-cell">
|
||||||
|
<div class="stat-num">300k+</div>
|
||||||
|
<div class="stat-label">Composer-Pakete</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="divider"></div>
|
||||||
|
|
||||||
|
<!-- MAIN GRID -->
|
||||||
|
<main class="grid">
|
||||||
|
|
||||||
|
<!-- ===== SPALTE 1 ===== -->
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<polygon points="12,2 22,8 22,16 12,22 2,16 2,8"/>
|
||||||
|
</svg>
|
||||||
|
Kernkonzepte
|
||||||
|
</h2>
|
||||||
|
<div class="feature-list">
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">1</div>
|
||||||
|
<div class="text"><b>Server-seitig</b> – läuft auf Webservern, generiert HTML pro Request</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">2</div>
|
||||||
|
<div class="text"><b>Dynamisch typisiert</b> – Typen zur Laufzeit, optional mit Type-Hints prüfbar</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">3</div>
|
||||||
|
<div class="text"><b>OOP & funktional</b> – Klassen, Interfaces, Traits, Enums, First-Class-Funktionen</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">4</div>
|
||||||
|
<div class="text"><b>Composer</b> – moderner Paket-Manager für Dependencies und Autoloading</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<path d="M3 6h18M3 12h18M3 18h18"/>
|
||||||
|
</svg>
|
||||||
|
Datentypen
|
||||||
|
</h2>
|
||||||
|
<div class="types">
|
||||||
|
<div class="type"><b>int</b><span>Ganze Zahlen</span></div>
|
||||||
|
<div class="type"><b>float</b><span>Kommazahlen</span></div>
|
||||||
|
<div class="type"><b>string</b><span>Text in '' oder ""</span></div>
|
||||||
|
<div class="type"><b>bool</b><span>true / false</span></div>
|
||||||
|
<div class="type"><b>array</b><span>Liste oder Map</span></div>
|
||||||
|
<div class="type"><b>object</b><span>Klassen-Instanz</span></div>
|
||||||
|
<div class="type"><b>null</b><span>kein Wert</span></div>
|
||||||
|
<div class="type"><b>callable</b><span>Funktion / Closure</span></div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ===== SPALTE 2 ===== -->
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<polyline points="16,18 22,12 16,6"/><polyline points="8,6 2,12 8,18"/>
|
||||||
|
</svg>
|
||||||
|
Hello World
|
||||||
|
</h2>
|
||||||
|
<div class="code"><span class="t"><?php</span><br>
|
||||||
|
<span class="k">declare</span>(strict_types=<span class="s">1</span>);<br>
|
||||||
|
<br>
|
||||||
|
<span class="k">function</span> <span class="f">greet</span>(<span class="t">string</span> <span class="v">$name</span>): <span class="t">string</span> {<br>
|
||||||
|
<span class="k">return</span> <span class="s">"Hallo, </span><span class="v">$name</span><span class="s">!"</span>;<br>
|
||||||
|
}<br>
|
||||||
|
<br>
|
||||||
|
<span class="k">echo</span> <span class="f">greet</span>(<span class="s">'Marek'</span>);<br>
|
||||||
|
<span class="c">// Hallo, Marek!</span></div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<path d="M22 12h-4l-3 9L9 3l-3 9H2"/>
|
||||||
|
</svg>
|
||||||
|
Moderne vs. Legacy
|
||||||
|
</h2>
|
||||||
|
<div class="split">
|
||||||
|
<div class="split-col modern">
|
||||||
|
<h4>Modern (8.x)</h4>
|
||||||
|
<ul>
|
||||||
|
<li>strict_types</li>
|
||||||
|
<li>Readonly Properties</li>
|
||||||
|
<li>Enums & Match</li>
|
||||||
|
<li>Named Arguments</li>
|
||||||
|
<li>Promoted Constructor</li>
|
||||||
|
<li>Composer + PSR</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="split-col legacy">
|
||||||
|
<h4>Legacy (5.x)</h4>
|
||||||
|
<ul>
|
||||||
|
<li>magic_quotes</li>
|
||||||
|
<li>register_globals</li>
|
||||||
|
<li>mysql_* Funktionen</li>
|
||||||
|
<li>kein Typ-System</li>
|
||||||
|
<li>include/require Chaos</li>
|
||||||
|
<li>Spaghetti-Code</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ===== SPALTE 3 ===== -->
|
||||||
|
<div class="col">
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<circle cx="12" cy="12" r="3"/><circle cx="12" cy="12" r="9"/>
|
||||||
|
<line x1="12" y1="3" x2="12" y2="9"/><line x1="12" y1="15" x2="12" y2="21"/>
|
||||||
|
<line x1="3" y1="12" x2="9" y2="12"/><line x1="15" y1="12" x2="21" y2="12"/>
|
||||||
|
</svg>
|
||||||
|
Ökosystem
|
||||||
|
</h2>
|
||||||
|
<div class="eco-grid">
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">Laravel</span>
|
||||||
|
<span class="eco-desc">Full-Stack Framework</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">Symfony</span>
|
||||||
|
<span class="eco-desc">Enterprise-Framework</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">Composer</span>
|
||||||
|
<span class="eco-desc">Paket-Manager</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">PHPUnit</span>
|
||||||
|
<span class="eco-desc">Testing-Framework</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">PHPStan</span>
|
||||||
|
<span class="eco-desc">Static Analysis</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="eco-item">
|
||||||
|
<div class="eco-ico"><svg width="2.8mm" height="2.8mm" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="3"><polyline points="20,6 9,17 4,12"/></svg></div>
|
||||||
|
<div class="eco-text">
|
||||||
|
<span class="eco-name">Shopware</span>
|
||||||
|
<span class="eco-desc">E-Commerce-Plattform</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="block">
|
||||||
|
<h2>
|
||||||
|
<svg class="ico" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<path d="M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5"/>
|
||||||
|
</svg>
|
||||||
|
Einsatzgebiete
|
||||||
|
</h2>
|
||||||
|
<div class="feature-list">
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">W</div>
|
||||||
|
<div class="text"><b>Web-Backends</b> – REST/GraphQL APIs, MVC-Apps mit Laravel oder Symfony</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">C</div>
|
||||||
|
<div class="text"><b>CMS</b> – WordPress, Drupal, TYPO3 für Content-Management</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">E</div>
|
||||||
|
<div class="text"><b>E-Commerce</b> – Shopware, Magento, WooCommerce</div>
|
||||||
|
</div>
|
||||||
|
<div class="feature-row">
|
||||||
|
<div class="dot">$</div>
|
||||||
|
<div class="text"><b>CLI-Tools</b> – Symfony Console, Laravel Artisan für Automation</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<!-- FOOTER -->
|
||||||
|
<footer class="footer">
|
||||||
|
<div class="footer-text">
|
||||||
|
<b>Wann PHP wählen?</b> Wenn du schnell ein Web-Backend brauchst, ein riesiges Ökosystem nutzen willst und eine reife Sprache mit großem Talent-Pool suchst. Moderne PHP 8 ist nicht das PHP von 2005.
|
||||||
|
</div>
|
||||||
|
<div class="footer-tag">php.net</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
```
|
||||||
Reference in New Issue
Block a user