{
"event": "PreToolUse",
"tool_name": "Edit",
"tool_input": {
"file_path": "\/var\/www\/dev.campus.systemische-tools.de\/docs\/plan-form-components-refactoring.md",
"old_string": "### 0e.3 Presets\n\n**3 feste Presets:**\n\n| Preset | Temperature | Max Tokens | Beschreibung |\n|--------|-------------|------------|--------------|\n| **Präzise** | 0.3 | 50% | Fokussiert, weniger kreativ |\n| **Ausgewogen** | 0.7 | 75% | Standard für die meisten Aufgaben |\n| **Kreativ** | 0.9 | 100% | Explorativ, längere Antworten |\n\n**User-Präferenz:** `user_preferences.default_preset`\n\n```sql\n-- Erweiterung user_preferences\nALTER TABLE user_preferences\nADD COLUMN default_preset ENUM('precise', 'balanced', 'creative') DEFAULT 'balanced';\n```",
"new_string": "### 0e.3 Presets (DB-verwaltet unter `\/ressourcen\/llm`)\n\n**Gewählt: Option B - Presets in DB, verwaltbar**\n\n#### Tabelle `llm_presets`\n\n```sql\nCREATE TABLE llm_presets (\n id INT AUTO_INCREMENT PRIMARY KEY,\n slug VARCHAR(50) NOT NULL UNIQUE, -- \"precise\", \"balanced\", \"creative\"\n display_name VARCHAR(100) NOT NULL, -- \"Präzise\", \"Ausgewogen\", \"Kreativ\"\n description TEXT, -- \"Fokussiert, weniger kreativ\"\n temperature DECIMAL(3,2) NOT NULL, -- 0.30\n tokens_percent INT NOT NULL, -- 50 (= 50% von max_output_tokens)\n icon VARCHAR(50), -- Optional: Icon-Klasse\n is_active BOOLEAN DEFAULT TRUE,\n sort_order INT DEFAULT 0,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n\n-- Initial-Daten\nINSERT INTO llm_presets (slug, display_name, description, temperature, tokens_percent, sort_order) VALUES\n('precise', 'Präzise', 'Fokussiert, weniger kreativ', 0.30, 50, 1),\n('balanced', 'Ausgewogen', 'Standard für die meisten Aufgaben', 0.70, 75, 2),\n('creative', 'Kreativ', 'Explorativ, längere Antworten', 0.90, 100, 3);\n```\n\n#### Admin-Seite `\/ressourcen\/llm` erweitern\n\n```\n┌─────────────────────────────────────────────────────┐\n│ LLM-Verwaltung │\n├─────────────────────────────────────────────────────┤\n│ │\n│ [Tab: Modelle] [Tab: Presets] │\n│ │\n│ ═══════════════════════════════════════════════════ │\n│ │\n│ Presets [+ Neues Preset]\n│ ┌──────────────────────────────────────────────────┐\n│ │ Name │ Temp │ Tokens │ Aktiv │ Aktionen │\n│ ├──────────────────────────────────────────────────┤\n│ │ Präzise │ 0.3 │ 50% │ ✓ │ Bearbeiten│\n│ │ Ausgewogen │ 0.7 │ 75% │ ✓ │ Bearbeiten│\n│ │ Kreativ │ 0.9 │ 100% │ ✓ │ Bearbeiten│\n│ └──────────────────────────────────────────────────┘\n│ │\n└─────────────────────────────────────────────────────┘\n```\n\n#### User-Präferenz\n\n```sql\n-- In user_preferences:\nALTER TABLE user_preferences\nADD COLUMN default_preset_id INT,\nADD CONSTRAINT fk_prefs_preset FOREIGN KEY (default_preset_id) REFERENCES llm_presets(id);\n```\n\n#### Repository\n\n```php\n\/\/ LlmPresetRepository.php\nclass LlmPresetRepository\n{\n public function getActive(): array\n {\n return $this->db->query(\"\n SELECT * FROM llm_presets\n WHERE is_active = 1\n ORDER BY sort_order ASC\n \")->fetchAll();\n }\n\n public function findBySlug(string $slug): ?array\n {\n \/\/ ...\n }\n}\n```"
}
}
{
"tool_response": {
"filePath": "\/var\/www\/dev.campus.systemische-tools.de\/docs\/plan-form-components-refactoring.md",
"oldString": "### 0e.3 Presets\n\n**3 feste Presets:**\n\n| Preset | Temperature | Max Tokens | Beschreibung |\n|--------|-------------|------------|--------------|\n| **Präzise** | 0.3 | 50% | Fokussiert, weniger kreativ |\n| **Ausgewogen** | 0.7 | 75% | Standard für die meisten Aufgaben |\n| **Kreativ** | 0.9 | 100% | Explorativ, längere Antworten |\n\n**User-Präferenz:** `user_preferences.default_preset`\n\n```sql\n-- Erweiterung user_preferences\nALTER TABLE user_preferences\nADD COLUMN default_preset ENUM('precise', 'balanced', 'creative') DEFAULT 'balanced';\n```",
"newString": "### 0e.3 Presets (DB-verwaltet unter `\/ressourcen\/llm`)\n\n**Gewählt: Option B - Presets in DB, verwaltbar**\n\n#### Tabelle `llm_presets`\n\n```sql\nCREATE TABLE llm_presets (\n id INT AUTO_INCREMENT PRIMARY KEY,\n slug VARCHAR(50) NOT NULL UNIQUE, -- \"precise\", \"balanced\", \"creative\"\n display_name VARCHAR(100) NOT NULL, -- \"Präzise\", \"Ausgewogen\", \"Kreativ\"\n description TEXT, -- \"Fokussiert, weniger kreativ\"\n temperature DECIMAL(3,2) NOT NULL, -- 0.30\n tokens_percent INT NOT NULL, -- 50 (= 50% von max_output_tokens)\n icon VARCHAR(50), -- Optional: Icon-Klasse\n is_active BOOLEAN DEFAULT TRUE,\n sort_order INT DEFAULT 0,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n\n-- Initial-Daten\nINSERT INTO llm_presets (slug, display_name, description, temperature, tokens_percent, sort_order) VALUES\n('precise', 'Präzise', 'Fokussiert, weniger kreativ', 0.30, 50, 1),\n('balanced', 'Ausgewogen', 'Standard für die meisten Aufgaben', 0.70, 75, 2),\n('creative', 'Kreativ', 'Explorativ, längere Antworten', 0.90, 100, 3);\n```\n\n#### Admin-Seite `\/ressourcen\/llm` erweitern\n\n```\n┌─────────────────────────────────────────────────────┐\n│ LLM-Verwaltung │\n├─────────────────────────────────────────────────────┤\n│ │\n│ [Tab: Modelle] [Tab: Presets] │\n│ │\n│ ═══════════════════════════════════════════════════ │\n│ │\n│ Presets [+ Neues Preset]\n│ ┌──────────────────────────────────────────────────┐\n│ │ Name │ Temp │ Tokens │ Aktiv │ Aktionen │\n│ ├──────────────────────────────────────────────────┤\n│ │ Präzise │ 0.3 │ 50% │ ✓ │ Bearbeiten│\n│ │ Ausgewogen │ 0.7 │ 75% │ ✓ │ Bearbeiten│\n│ │ Kreativ │ 0.9 │ 100% │ ✓ │ Bearbeiten│\n│ └──────────────────────────────────────────────────┘\n│ │\n└─────────────────────────────────────────────────────┘\n```\n\n#### User-Präferenz\n\n```sql\n-- In user_preferences:\nALTER TABLE user_preferences\nADD COLUMN default_preset_id INT,\nADD CONSTRAINT fk_prefs_preset FOREIGN KEY (default_preset_id) REFERENCES llm_presets(id);\n```\n\n#### Repository\n\n```php\n\/\/ LlmPresetRepository.php\nclass LlmPresetRepository\n{\n public function getActive(): array\n {\n return $this->db->query(\"\n SELECT * FROM llm_presets\n WHERE is_active = 1\n ORDER BY sort_order ASC\n \")->fetchAll();\n }\n\n public function findBySlug(string $slug): ?array\n {\n \/\/ ...\n }\n}\n```",
"originalFile": "# Plan: Einheitliche Form-Komponenten (Chat & Content Studio)\n\n## Status: PLANUNG\n**Erstellt:** 2025-12-21\n**Ziel:** DRY, KISS, SRP - Shared Partials für alle wiederverwendbaren Form-Elemente\n\n---\n\n## 0. LLM-Verwaltung (Grundlage für Modell-Dropdown)\n\n### 0.1 Architektur-Entscheidung\n\n**Gewählt: Variante B - Dedizierte LLM-Verwaltung**\n\nStatt dynamischer API-Abfrage bei jedem Seitenaufruf:\n- Zentrale Admin-Seite `\/ressourcen\/llm`\n- Datenbank-Tabelle als Single Source of Truth\n- Sync-Buttons für Provider (Anthropic, Ollama, weitere)\n- Sprechende Namen und Zusatzmetadaten\n\n### 0.2 Datenbank-Tabelle `llm_models`\n\n```sql\nCREATE TABLE llm_models (\n id INT AUTO_INCREMENT PRIMARY KEY,\n provider ENUM('anthropic', 'ollama', 'openai', 'google', 'mistral', 'custom') NOT NULL,\n model_id VARCHAR(100) NOT NULL, -- API-ID: \"claude-opus-4-5-20251101\"\n display_name VARCHAR(100) NOT NULL, -- Anzeige: \"Claude Opus 4.5\"\n description TEXT, -- Kurzbeschreibung\n context_window INT, -- Max. Tokens Input: 200000\n max_output_tokens INT, -- Max. Tokens Output: 8192\n input_price_per_mtok DECIMAL(10,4), -- Preis Input $\/MTok\n output_price_per_mtok DECIMAL(10,4), -- Preis Output $\/MTok\n capabilities JSON, -- {\"vision\": true, \"function_calling\": true}\n is_local BOOLEAN DEFAULT FALSE, -- Lokal (Ollama) vs. Cloud\n is_active BOOLEAN DEFAULT TRUE, -- In Dropdowns anzeigen?\n sort_order INT DEFAULT 0, -- Reihenfolge in Dropdowns\n last_synced_at DATETIME, -- Letzte Synchronisation\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,\n UNIQUE KEY unique_provider_model (provider, model_id)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n```\n\n### 0.3 Admin-Seite `\/ressourcen\/llm`\n\n**URL:** `\/ressourcen\/llm`\n**Controller:** `RessourcenController::llmIndex()`\n\n**Funktionen:**\n\n| Button | Aktion |\n|--------|--------|\n| \"Anthropic synchronisieren\" | `GET api.anthropic.com\/v1\/models` → DB aktualisieren |\n| \"Ollama synchronisieren\" | `ollama list` → DB aktualisieren |\n| \"Neuer Provider\" | Manuell weiteren Provider hinzufügen |\n\n**Tabellen-Ansicht:**\n\n| Provider | Model-ID | Display-Name | Context | Preis In\/Out | Lokal | Aktiv | Aktionen |\n|----------|----------|--------------|---------|--------------|-------|-------|----------|\n| Anthropic | claude-opus-4-5-20251101 | Claude Opus 4.5 | 200k | $15\/$75 | - | ✓ | Bearbeiten |\n| Ollama | mistral:latest | Mistral 7B | 32k | - | ✓ | ✓ | Bearbeiten |\n\n**Bearbeiten-Dialog:**\n- Display-Name ändern\n- Beschreibung hinzufügen\n- Context-Window \/ Max-Output korrigieren\n- Preise eintragen (für Kostenberechnung)\n- Aktivieren\/Deaktivieren\n- Sortierung ändern\n\n### 0.4 Sync-Logik\n\n**Anthropic Sync:**\n```php\n\/\/ GET https:\/\/api.anthropic.com\/v1\/models\n\/\/ Response: {\"data\": [{\"id\": \"claude-opus-4-5-20251101\", ...}]}\n\nforeach ($apiModels as $model) {\n \/\/ INSERT ... ON DUPLICATE KEY UPDATE\n \/\/ Neue Modelle: is_active = true, display_name = model_id (initial)\n \/\/ Existierende: last_synced_at aktualisieren\n \/\/ Fehlende: NICHT löschen, nur last_synced_at bleibt alt\n}\n```\n\n**Ollama Sync:**\n```php\n\/\/ $ ollama list\n\/\/ NAME ID SIZE MODIFIED\n\/\/ mistral:latest abc123... 4.1 GB 2 days ago\n\n$output = shell_exec('ollama list');\n\/\/ Parsen und in DB einfügen\n```\n\n### 0.5 Dropdown-Query\n\n```php\n\/\/ ModelService::getActiveModels()\npublic function getActiveModels(): array\n{\n return $this->db->query(\"\n SELECT model_id, display_name, provider, is_local, context_window\n FROM llm_models\n WHERE is_active = 1\n ORDER BY is_local ASC, sort_order ASC, display_name ASC\n \")->fetchAll();\n}\n```\n\n### 0.6 Partial nutzt DB-Daten\n\n```php\n\/\/ \/src\/View\/partials\/form\/model-select.php\n<?php\n$models = $models ?? [];\n$selected = $selected ?? '';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"model\" id=\"model\" class=\"<?= $class ?>\">\n <optgroup label=\"Cloud\">\n <?php foreach ($models as $m): ?>\n <?php if (!$m['is_local']): ?>\n <option value=\"<?= $m['model_id'] ?>\" <?= $selected === $m['model_id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($m['display_name']) ?>\n <\/option>\n <?php endif; ?>\n <?php endforeach; ?>\n <\/optgroup>\n <optgroup label=\"Lokal\">\n <?php foreach ($models as $m): ?>\n <?php if ($m['is_local']): ?>\n <option value=\"<?= $m['model_id'] ?>\" <?= $selected === $m['model_id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($m['display_name']) ?>\n <\/option>\n <?php endif; ?>\n <?php endforeach; ?>\n <\/optgroup>\n<\/select>\n```\n\n### 0.7 Migrations-Schritte (LLM-Verwaltung)\n\n1. [ ] Tabelle `llm_models` in `ki_dev` erstellen\n2. [ ] `LlmModelRepository` erstellen\n3. [ ] `LlmSyncService` erstellen (Anthropic + Ollama)\n4. [ ] Route `\/ressourcen\/llm` anlegen\n5. [ ] `RessourcenController::llmIndex()` implementieren\n6. [ ] View `\/ressourcen\/llm\/index.php` erstellen\n7. [ ] Sync-Buttons implementieren\n8. [ ] Bearbeiten-Funktionalität\n9. [ ] Initial-Sync durchführen (bestehende Modelle importieren)\n10. [ ] `ModelConfig.php` durch `LlmModelRepository` ersetzen\n11. [ ] Partial `model-select.php` erstellen\n12. [ ] Chat und Content auf Partial umstellen\n\n---\n\n## 0b. Collection-Verwaltung (Grundlage für Collection-Dropdown)\n\n### 0b.1 Architektur-Entscheidung\n\n**Gewählt: DB-Tabelle analog zu LLMs**\n\n- Zentrale Admin-Seite `\/ressourcen\/collections`\n- Datenbank-Tabelle als Single Source of Truth\n- Sync mit Qdrant (Metadaten abrufen)\n- Sprechende Namen und Beschreibungen\n- **Multi-Select** als einheitliche Darstellung\n\n### 0b.2 Datenbank-Tabelle `rag_collections`\n\n```sql\nCREATE TABLE rag_collections (\n id INT AUTO_INCREMENT PRIMARY KEY,\n collection_id VARCHAR(100) NOT NULL UNIQUE, -- Qdrant-Name: \"documents\"\n display_name VARCHAR(100) NOT NULL, -- Anzeige: \"Dokumente\"\n description TEXT, -- \"PDF-Dokumente aus Nextcloud\"\n\n -- Qdrant-Metadaten (via Sync)\n vector_size INT, -- 1024\n distance_metric VARCHAR(20), -- \"Cosine\"\n points_count INT DEFAULT 0, -- Anzahl Vektoren\n\n -- Konfiguration\n embedding_model VARCHAR(100), -- \"mxbai-embed-large\"\n chunk_size INT, -- 2000\n chunk_overlap INT, -- 200\n\n -- Verwaltung\n source_type ENUM('nextcloud', 'mail', 'manual', 'system') DEFAULT 'manual',\n source_path VARCHAR(500), -- \"\/var\/www\/nextcloud\/data\/...\"\n is_active BOOLEAN DEFAULT TRUE, -- In Dropdowns anzeigen?\n is_searchable BOOLEAN DEFAULT TRUE, -- Für RAG verfügbar?\n sort_order INT DEFAULT 0,\n\n -- Timestamps\n last_synced_at DATETIME,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n```\n\n### 0b.3 Impact-Analyse: Wer nutzt Collections?\n\n| System\/Seite | Aktueller Zugriff | Neuer Zugriff | Änderung |\n|--------------|-------------------|---------------|----------|\n| **Chat** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Content Studio** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Content Show** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Pipeline (Python)** | `config.py QDRANT_COLLECTIONS` | DB-Query oder Config-Sync | Python liest DB |\n| **Semantic Explorer** | Direkt Qdrant | `CollectionRepository` + Qdrant | Metadaten aus DB |\n| **System Explorer** | - | Statistiken aus DB | Neu |\n| **API \/api\/v1\/search** | Qdrant direkt | Validierung gegen DB | Nur aktive Collections |\n| **Embedding-Service** | `config.py` | DB oder Sync | Konsistenz |\n\n### 0b.4 Admin-Seite `\/ressourcen\/collections`\n\n**URL:** `\/ressourcen\/collections`\n**Controller:** `RessourcenController::collectionsIndex()`\n\n**Funktionen:**\n\n| Button | Aktion |\n|--------|--------|\n| \"Qdrant synchronisieren\" | Collections + Metadaten von Qdrant abrufen |\n| \"Neue Collection\" | Manuell Collection registrieren |\n\n**Tabellen-Ansicht:**\n\n| Collection-ID | Display-Name | Vektoren | Größe | Quelle | Aktiv | Suchbar | Aktionen |\n|---------------|--------------|----------|-------|--------|-------|---------|----------|\n| documents | Dokumente | 1.247 | 1024d | Nextcloud | ✓ | ✓ | Bearbeiten |\n| dokumentation_chunks | Doku-Chunks | 892 | 1024d | System | ✓ | ✓ | Bearbeiten |\n| mail | E-Mails | 3.421 | 1024d | Mail | ✓ | ✓ | Bearbeiten |\n| entities | Entitäten | 156 | 1024d | System | ✓ | - | Bearbeiten |\n\n**Bearbeiten-Dialog:**\n- Display-Name ändern\n- Beschreibung hinzufügen\n- Quelle\/Pfad dokumentieren\n- Aktivieren\/Deaktivieren (Dropdown)\n- Suchbar ja\/nein (RAG-Verfügbarkeit)\n- Sortierung ändern\n\n### 0b.5 Sync-Logik\n\n**Qdrant Sync:**\n```php\n\/\/ QdrantService::getCollectionsInfo()\npublic function syncCollections(): array\n{\n $qdrantCollections = $this->client->getCollections();\n\n foreach ($qdrantCollections as $col) {\n $info = $this->client->getCollectionInfo($col['name']);\n\n \/\/ INSERT ... ON DUPLICATE KEY UPDATE\n $this->repository->upsert([\n 'collection_id' => $col['name'],\n 'display_name' => $col['name'], \/\/ Initial = ID\n 'vector_size' => $info['config']['params']['size'],\n 'distance_metric' => $info['config']['params']['distance'],\n 'points_count' => $info['points_count'],\n 'last_synced_at' => now(),\n ]);\n }\n}\n```\n\n### 0b.6 Repository-Interface\n\n```php\n\/\/ CollectionRepository.php\ninterface CollectionRepositoryInterfac... [TRUNCATED-ec314a2fbd612d6d]",
"structuredPatch": [
{
"oldStart": 760,
"oldLines": 24,
"newStart": 760,
"newLines": 88,
"lines": [
" \/\/ → [1024, 2048, 3072, 4096]",
" ```",
" ",
"-### 0e.3 Presets",
"+### 0e.3 Presets (DB-verwaltet unter `\/ressourcen\/llm`)",
" ",
"-**3 feste Presets:**",
"+**Gewählt: Option B - Presets in DB, verwaltbar**",
" ",
"-| Preset | Temperature | Max Tokens | Beschreibung |",
"-|--------|-------------|------------|--------------|",
"-| **Präzise** | 0.3 | 50% | Fokussiert, weniger kreativ |",
"-| **Ausgewogen** | 0.7 | 75% | Standard für die meisten Aufgaben |",
"-| **Kreativ** | 0.9 | 100% | Explorativ, längere Antworten |",
"+#### Tabelle `llm_presets`",
" ",
"-**User-Präferenz:** `user_preferences.default_preset`",
"+```sql",
"+CREATE TABLE llm_presets (",
"+ id INT AUTO_INCREMENT PRIMARY KEY,",
"+ slug VARCHAR(50) NOT NULL UNIQUE, -- \"precise\", \"balanced\", \"creative\"",
"+ display_name VARCHAR(100) NOT NULL, -- \"Präzise\", \"Ausgewogen\", \"Kreativ\"",
"+ description TEXT, -- \"Fokussiert, weniger kreativ\"",
"+ temperature DECIMAL(3,2) NOT NULL, -- 0.30",
"+ tokens_percent INT NOT NULL, -- 50 (= 50% von max_output_tokens)",
"+ icon VARCHAR(50), -- Optional: Icon-Klasse",
"+ is_active BOOLEAN DEFAULT TRUE,",
"+ sort_order INT DEFAULT 0,",
"+ created_at DATETIME DEFAULT CURRENT_TIMESTAMP,",
"+ updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP",
"+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;",
" ",
"+-- Initial-Daten",
"+INSERT INTO llm_presets (slug, display_name, description, temperature, tokens_percent, sort_order) VALUES",
"+('precise', 'Präzise', 'Fokussiert, weniger kreativ', 0.30, 50, 1),",
"+('balanced', 'Ausgewogen', 'Standard für die meisten Aufgaben', 0.70, 75, 2),",
"+('creative', 'Kreativ', 'Explorativ, längere Antworten', 0.90, 100, 3);",
"+```",
"+",
"+#### Admin-Seite `\/ressourcen\/llm` erweitern",
"+",
"+```",
"+┌─────────────────────────────────────────────────────┐",
"+│ LLM-Verwaltung │",
"+├─────────────────────────────────────────────────────┤",
"+│ │",
"+│ [Tab: Modelle] [Tab: Presets] │",
"+│ │",
"+│ ═══════════════════════════════════════════════════ │",
"+│ │",
"+│ Presets [+ Neues Preset]",
"+│ ┌──────────────────────────────────────────────────┐",
"+│ │ Name │ Temp │ Tokens │ Aktiv │ Aktionen │",
"+│ ├──────────────────────────────────────────────────┤",
"+│ │ Präzise │ 0.3 │ 50% │ ✓ │ Bearbeiten│",
"+│ │ Ausgewogen │ 0.7 │ 75% │ ✓ │ Bearbeiten│",
"+│ │ Kreativ │ 0.9 │ 100% │ ✓ │ Bearbeiten│",
"+│ └──────────────────────────────────────────────────┘",
"+│ │",
"+└─────────────────────────────────────────────────────┘",
"+```",
"+",
"+#### User-Präferenz",
"+",
" ```sql",
"--- Erweiterung user_preferences",
"+-- In user_preferences:",
" ALTER TABLE user_preferences",
"-ADD COLUMN default_preset ENUM('precise', 'balanced', 'creative') DEFAULT 'balanced';",
"+ADD COLUMN default_preset_id INT,",
"+ADD CONSTRAINT fk_prefs_preset FOREIGN KEY (default_preset_id) REFERENCES llm_presets(id);",
" ```",
" ",
"+#### Repository",
"+",
"+```php",
"+\/\/ LlmPresetRepository.php",
"+class LlmPresetRepository",
"+{",
"+ public function getActive(): array",
"+ {",
"+ return $this->db->query(\"",
"+ SELECT * FROM llm_presets",
"+ WHERE is_active = 1",
"+ ORDER BY sort_order ASC",
"+ \")->fetchAll();",
"+ }",
"+",
"+ public function findBySlug(string $slug): ?array",
"+ {",
"+ \/\/ ...",
"+ }",
"+}",
"+```",
"+",
" ### 0e.4 Partial: temperature-tokens.php",
" ",
" ```php"
]
}
],
"userModified": false,
"replaceAll": false
}
}