This repository has been archived on 2026-02-15. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
Benjamin Admin 21a844cb8a fix: Restore all files lost during destructive rebase
A previous `git pull --rebase origin main` dropped 177 local commits,
losing 3400+ files across admin-v2, backend, studio-v2, website,
klausur-service, and many other services. The partial restore attempt
(660295e2) only recovered some files.

This commit restores all missing files from pre-rebase ref 98933f5e
while preserving post-rebase additions (night-scheduler, night-mode UI,
NightModeWidget dashboard integration).

Restored features include:
- AI Module Sidebar (FAB), OCR Labeling, OCR Compare
- GPU Dashboard, RAG Pipeline, Magic Help
- Klausur-Korrektur (8 files), Abitur-Archiv (5+ files)
- Companion, Zeugnisse-Crawler, Screen Flow
- Full backend, studio-v2, website, klausur-service
- All compliance SDKs, agent-core, voice-service
- CI/CD configs, documentation, scripts

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-09 09:51:32 +01:00

955 lines
24 KiB
Python

"""
BreakPilot Studio - Dashboard/Startansicht Modul
Funktionen:
- Startansicht mit Kacheln zu allen Modulen
- Schnellzugriff auf die wichtigsten Funktionen
- Uebersicht ueber aktuelle Aktivitaeten
"""
class DashboardModule:
"""Modul fuer die Startansicht/Dashboard."""
@staticmethod
def get_css() -> str:
"""CSS fuer das Dashboard-Modul."""
return """
/* =============================================
DASHBOARD MODULE - Startansicht
============================================= */
/* Panel Layout */
.panel-dashboard {
display: flex;
flex-direction: column;
height: 100%;
background: var(--bp-bg);
overflow-y: auto;
}
/* Dashboard Header */
.dashboard-header {
padding: 32px 40px 24px;
background: var(--bp-surface);
border-bottom: 1px solid var(--bp-border);
}
.dashboard-welcome {
margin-bottom: 8px;
}
.dashboard-welcome h1 {
font-size: 28px;
font-weight: 700;
color: var(--bp-text);
margin: 0;
}
.dashboard-welcome p {
font-size: 14px;
color: var(--bp-text-muted);
margin-top: 8px;
}
/* Dashboard Content */
.dashboard-content {
padding: 32px 40px;
flex: 1;
}
/* Section Titles */
.dashboard-section-title {
font-size: 14px;
font-weight: 600;
color: var(--bp-text-muted);
text-transform: uppercase;
letter-spacing: 0.5px;
margin-bottom: 20px;
}
/* Module Cards Grid */
.dashboard-cards {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
gap: 24px;
margin-bottom: 48px;
}
/* Module Card */
.dashboard-card {
background: var(--bp-surface);
border: 1px solid var(--bp-border);
border-radius: 16px;
padding: 24px;
cursor: pointer;
transition: all 0.3s ease;
position: relative;
overflow: hidden;
}
.dashboard-card:hover {
transform: translateY(-4px);
box-shadow: 0 12px 24px rgba(0, 0, 0, 0.15);
border-color: var(--bp-primary);
}
.dashboard-card::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
height: 4px;
background: var(--bp-primary);
opacity: 0;
transition: opacity 0.3s;
}
.dashboard-card:hover::before {
opacity: 1;
}
.dashboard-card-icon {
width: 56px;
height: 56px;
border-radius: 14px;
display: flex;
align-items: center;
justify-content: center;
font-size: 28px;
margin-bottom: 16px;
background: var(--bp-primary-soft);
}
.dashboard-card-title {
font-size: 18px;
font-weight: 600;
color: var(--bp-text);
margin-bottom: 8px;
}
.dashboard-card-description {
font-size: 13px;
color: var(--bp-text-muted);
line-height: 1.5;
margin-bottom: 16px;
}
.dashboard-card-footer {
display: flex;
justify-content: space-between;
align-items: center;
padding-top: 16px;
border-top: 1px solid var(--bp-border-subtle);
}
.dashboard-card-action {
font-size: 13px;
font-weight: 500;
color: var(--bp-primary);
display: flex;
align-items: center;
gap: 6px;
}
.dashboard-card-action::after {
content: '';
transition: transform 0.2s;
}
.dashboard-card:hover .dashboard-card-action::after {
transform: translateX(4px);
}
.dashboard-card-badge {
font-size: 11px;
padding: 4px 10px;
border-radius: 12px;
background: var(--bp-accent-soft);
color: var(--bp-accent);
font-weight: 500;
}
.dashboard-card-badge.new {
background: rgba(59, 130, 246, 0.1);
color: #3b82f6;
}
.dashboard-card-badge.beta {
background: rgba(245, 158, 11, 0.1);
color: #f59e0b;
}
/* Card Color Variants */
.dashboard-card.worksheets .dashboard-card-icon {
background: rgba(59, 130, 246, 0.1);
color: #3b82f6;
}
.dashboard-card.correction .dashboard-card-icon {
background: rgba(16, 185, 129, 0.1);
color: #10b981;
}
.dashboard-card.jitsi .dashboard-card-icon {
background: rgba(139, 92, 246, 0.1);
color: #8b5cf6;
}
.dashboard-card.letters .dashboard-card-icon {
background: rgba(236, 72, 153, 0.1);
color: #ec4899;
}
.dashboard-card.messenger .dashboard-card-icon {
background: rgba(20, 184, 166, 0.1);
color: #14b8a6;
}
.dashboard-card.klausur-korrektur .dashboard-card-icon {
background: rgba(168, 85, 247, 0.1);
color: #a855f7;
}
/* Quick Stats */
.dashboard-stats {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));
gap: 16px;
margin-bottom: 48px;
}
.dashboard-stat {
background: var(--bp-surface);
border: 1px solid var(--bp-border);
border-radius: 12px;
padding: 20px;
display: flex;
align-items: center;
gap: 16px;
}
.dashboard-stat-icon {
width: 48px;
height: 48px;
border-radius: 12px;
display: flex;
align-items: center;
justify-content: center;
font-size: 24px;
background: var(--bp-surface-elevated);
}
.dashboard-stat-content {
flex: 1;
}
.dashboard-stat-value {
font-size: 24px;
font-weight: 700;
color: var(--bp-text);
}
.dashboard-stat-label {
font-size: 12px;
color: var(--bp-text-muted);
}
/* Recent Activity */
.dashboard-activity {
background: var(--bp-surface);
border: 1px solid var(--bp-border);
border-radius: 16px;
padding: 24px;
}
.dashboard-activity-title {
font-size: 16px;
font-weight: 600;
margin-bottom: 20px;
color: var(--bp-text);
}
.dashboard-activity-list {
list-style: none;
padding: 0;
margin: 0;
}
.dashboard-activity-item {
display: flex;
align-items: center;
gap: 16px;
padding: 14px 0;
border-bottom: 1px solid var(--bp-border-subtle);
}
.dashboard-activity-item:last-child {
border-bottom: none;
}
.dashboard-activity-icon {
width: 36px;
height: 36px;
border-radius: 8px;
display: flex;
align-items: center;
justify-content: center;
font-size: 16px;
background: var(--bp-surface-elevated);
}
.dashboard-activity-content {
flex: 1;
}
.dashboard-activity-text {
font-size: 14px;
color: var(--bp-text);
}
.dashboard-activity-time {
font-size: 12px;
color: var(--bp-text-muted);
}
/* Empty State */
.dashboard-empty {
text-align: center;
padding: 48px;
color: var(--bp-text-muted);
}
.dashboard-empty-icon {
font-size: 48px;
margin-bottom: 16px;
opacity: 0.5;
}
/* =============================================
AI PROMPT INPUT
============================================= */
.dashboard-ai-prompt {
background: var(--bp-surface);
border: 1px solid var(--bp-border);
border-radius: 16px;
padding: 20px;
margin-bottom: 32px;
}
.dashboard-ai-prompt-header {
display: flex;
align-items: center;
gap: 12px;
margin-bottom: 16px;
}
.dashboard-ai-prompt-icon {
width: 40px;
height: 40px;
border-radius: 10px;
display: flex;
align-items: center;
justify-content: center;
font-size: 20px;
background: linear-gradient(135deg, #6C1B1B, #991b1b);
color: white;
}
.dashboard-ai-prompt-title {
font-size: 16px;
font-weight: 600;
color: var(--bp-text);
}
.dashboard-ai-prompt-subtitle {
font-size: 12px;
color: var(--bp-text-muted);
}
.dashboard-ai-prompt-input-container {
display: flex;
gap: 12px;
align-items: flex-end;
}
.dashboard-ai-prompt-input {
flex: 1;
min-height: 44px;
max-height: 120px;
padding: 12px 16px;
border-radius: 12px;
border: 1px solid var(--bp-border);
background: var(--bp-bg);
color: var(--bp-text);
font-size: 14px;
font-family: inherit;
resize: none;
outline: none;
transition: border-color 0.2s;
}
.dashboard-ai-prompt-input:focus {
border-color: var(--bp-primary);
}
.dashboard-ai-prompt-input::placeholder {
color: var(--bp-text-muted);
}
.dashboard-ai-prompt-send {
width: 44px;
height: 44px;
border-radius: 12px;
border: none;
background: linear-gradient(135deg, #6C1B1B, #991b1b);
color: white;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
font-size: 18px;
transition: all 0.2s;
}
.dashboard-ai-prompt-send:hover {
transform: scale(1.05);
box-shadow: 0 4px 12px rgba(108, 27, 27, 0.4);
}
.dashboard-ai-prompt-send:disabled {
opacity: 0.5;
cursor: not-allowed;
transform: none;
}
.dashboard-ai-prompt-send.loading {
animation: pulse 1s infinite;
}
@keyframes pulse {
0%, 100% { opacity: 1; }
50% { opacity: 0.6; }
}
/* AI Response */
.dashboard-ai-response {
margin-top: 16px;
padding: 16px;
background: var(--bp-bg);
border-radius: 12px;
border: 1px solid var(--bp-border-subtle);
display: none;
}
.dashboard-ai-response.active {
display: block;
}
.dashboard-ai-response-header {
display: flex;
align-items: center;
gap: 8px;
margin-bottom: 12px;
color: var(--bp-text-muted);
font-size: 12px;
}
.dashboard-ai-response-text {
color: var(--bp-text);
font-size: 14px;
line-height: 1.6;
white-space: pre-wrap;
}
.dashboard-ai-response-text code {
background: var(--bp-surface-elevated);
padding: 2px 6px;
border-radius: 4px;
font-family: monospace;
}
.dashboard-ai-response-text pre {
background: var(--bp-surface-elevated);
padding: 12px;
border-radius: 8px;
overflow-x: auto;
margin: 12px 0;
}
.dashboard-ai-response-text pre code {
background: transparent;
padding: 0;
}
/* Model Selector */
.dashboard-ai-model-selector {
display: flex;
align-items: center;
gap: 8px;
margin-top: 12px;
padding-top: 12px;
border-top: 1px solid var(--bp-border-subtle);
}
.dashboard-ai-model-label {
font-size: 12px;
color: var(--bp-text-muted);
}
.dashboard-ai-model-select {
padding: 6px 12px;
border-radius: 8px;
border: 1px solid var(--bp-border);
background: var(--bp-bg);
color: var(--bp-text);
font-size: 12px;
cursor: pointer;
}
.dashboard-ai-model-select:focus {
outline: none;
border-color: var(--bp-primary);
}
"""
@staticmethod
def get_html() -> str:
"""HTML fuer das Dashboard-Modul."""
return """
<!-- Dashboard Panel -->
<div id="panel-dashboard" class="panel-dashboard">
<!-- Header -->
<div class="dashboard-header">
<div class="dashboard-welcome">
<h1>Willkommen bei BreakPilot Studio</h1>
<p>Waehle ein Modul, um zu beginnen</p>
</div>
</div>
<!-- Content -->
<div class="dashboard-content">
<!-- AI Prompt Input -->
<div class="dashboard-ai-prompt">
<div class="dashboard-ai-prompt-header">
<div class="dashboard-ai-prompt-icon">🤖</div>
<div>
<div class="dashboard-ai-prompt-title">KI-Assistent</div>
<div class="dashboard-ai-prompt-subtitle">Fragen Sie Ihren lokalen Ollama-Assistenten</div>
</div>
</div>
<div class="dashboard-ai-prompt-input-container">
<textarea
id="ai-prompt-input"
class="dashboard-ai-prompt-input"
placeholder="Stellen Sie eine Frage... (z.B. 'Wie schreibe ich einen Elternbrief?' oder 'Erstelle mir einen Lückentext über Brüche')"
rows="1"
onkeydown="handleAiPromptKeydown(event)"
oninput="autoResizeTextarea(this)"
></textarea>
<button id="ai-prompt-send" class="dashboard-ai-prompt-send" onclick="sendAiPrompt()">
</button>
</div>
<div class="dashboard-ai-response" id="ai-response">
<div class="dashboard-ai-response-header">
<span>🤖</span>
<span id="ai-response-model">Antwort</span>
</div>
<div class="dashboard-ai-response-text" id="ai-response-text"></div>
</div>
<div class="dashboard-ai-model-selector">
<span class="dashboard-ai-model-label">Modell:</span>
<select id="ai-model-select" class="dashboard-ai-model-select">
<option value="llama3.2:latest">Llama 3.2 (Standard)</option>
<option value="mistral:latest">Mistral</option>
<option value="qwen2.5:7b">Qwen 2.5 (7B)</option>
<option value="deepseek-coder:latest">DeepSeek Coder</option>
</select>
</div>
</div>
<!-- Module Cards -->
<div class="dashboard-section-title">Module</div>
<div class="dashboard-cards">
<!-- Arbeitsblaetter Studio -->
<div class="dashboard-card worksheets" onclick="loadModule('worksheets')">
<div class="dashboard-card-icon">📝</div>
<div class="dashboard-card-title">Arbeitsblaetter Studio</div>
<div class="dashboard-card-description">
Arbeitsblaetter hochladen, neu aufbauen und in Lerneinheiten organisieren. Generiere Mindmaps, Multiple Choice Tests und mehr.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
</div>
</div>
<!-- Klausurkorrektur -->
<div class="dashboard-card correction" onclick="loadModule('correction')">
<div class="dashboard-card-icon">✅</div>
<div class="dashboard-card-title">Klausurkorrektur</div>
<div class="dashboard-card-description">
Klausuren hochladen und automatisch korrigieren lassen. OCR-Erkennung und AI-gestuetzte Bewertung.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
<span class="dashboard-card-badge new">NEU</span>
</div>
</div>
<!-- Videokonferenz -->
<div class="dashboard-card jitsi" onclick="loadModule('jitsi')">
<div class="dashboard-card-icon">🎥</div>
<div class="dashboard-card-title">Videokonferenz</div>
<div class="dashboard-card-description">
Elterngespraeche, Klassenkonferenzen und Schulungen per Video. Integrierte Jitsi-Konferenzen.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
</div>
</div>
<!-- Elternbriefe -->
<div class="dashboard-card letters" onclick="loadModule('letters')">
<div class="dashboard-card-icon">✉️</div>
<div class="dashboard-card-title">Elternbriefe</div>
<div class="dashboard-card-description">
Elternbriefe mit rechtssicherer Sprache verfassen. GFK-Analyse und Legal Assistant inklusive.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
</div>
</div>
<!-- Messenger -->
<div class="dashboard-card messenger" onclick="loadModule('messenger')">
<div class="dashboard-card-icon">💬</div>
<div class="dashboard-card-title">Messenger</div>
<div class="dashboard-card-description">
Kontakte verwalten, Nachrichten senden, CSV Import/Export. DSGVO-konforme Elternkommunikation mit Vorlagen.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
</div>
</div>
<!-- Abiturklausuren (15-Punkte-System) - External Microservice -->
<div class="dashboard-card klausur-korrektur" onclick="openKlausurService()">
<div class="dashboard-card-icon">🎓</div>
<div class="dashboard-card-title">Abiturklausuren</div>
<div class="dashboard-card-description">
Abitur-Klausuren mit 15-Punkte-System. NiBiS-Aufgaben, Erwartungshorizont, Gutachten und Fairness-Analyse.
</div>
<div class="dashboard-card-footer">
<span class="dashboard-card-action">Oeffnen</span>
<span class="dashboard-card-badge new">Neu</span>
</div>
</div>
</div>
<!-- Recent Activity -->
<div class="dashboard-section-title">Letzte Aktivitaeten</div>
<div class="dashboard-activity">
<ul class="dashboard-activity-list" id="dashboard-activity-list">
<li class="dashboard-activity-item">
<div class="dashboard-activity-icon">📝</div>
<div class="dashboard-activity-content">
<div class="dashboard-activity-text">Willkommen bei BreakPilot Studio!</div>
<div class="dashboard-activity-time">Gerade eben</div>
</div>
</li>
</ul>
</div>
</div>
</div>
"""
@staticmethod
def get_js() -> str:
"""JavaScript fuer das Dashboard-Modul."""
return """
// =============================================
// DASHBOARD MODULE - Startansicht
// =============================================
let dashboardInitialized = false;
function loadDashboardModule() {
if (dashboardInitialized) {
console.log('Dashboard module already initialized');
return;
}
console.log('Loading Dashboard Module...');
// Load recent activity from localStorage
loadRecentActivity();
dashboardInitialized = true;
console.log('Dashboard Module loaded successfully');
}
function loadRecentActivity() {
const activityList = document.getElementById('dashboard-activity-list');
if (!activityList) return;
// Get stored activity
const stored = localStorage.getItem('bp-activity');
if (!stored) return;
try {
const activities = JSON.parse(stored);
if (!activities.length) return;
activityList.innerHTML = activities.slice(0, 5).map(act => `
<li class="dashboard-activity-item">
<div class="dashboard-activity-icon">${act.icon || '📄'}</div>
<div class="dashboard-activity-content">
<div class="dashboard-activity-text">${act.text}</div>
<div class="dashboard-activity-time">${formatActivityTime(act.time)}</div>
</div>
</li>
`).join('');
} catch (e) {
console.error('Error loading activity:', e);
}
}
function addActivity(icon, text) {
const stored = localStorage.getItem('bp-activity');
let activities = [];
try {
activities = stored ? JSON.parse(stored) : [];
} catch (e) {}
activities.unshift({
icon: icon,
text: text,
time: Date.now()
});
// Keep only last 20
activities = activities.slice(0, 20);
localStorage.setItem('bp-activity', JSON.stringify(activities));
}
function formatActivityTime(timestamp) {
const diff = Date.now() - timestamp;
const minutes = Math.floor(diff / 60000);
const hours = Math.floor(diff / 3600000);
const days = Math.floor(diff / 86400000);
if (minutes < 1) return 'Gerade eben';
if (minutes < 60) return `Vor ${minutes} Minute${minutes > 1 ? 'n' : ''}`;
if (hours < 24) return `Vor ${hours} Stunde${hours > 1 ? 'n' : ''}`;
return `Vor ${days} Tag${days > 1 ? 'en' : ''}`;
}
// Show Dashboard Panel
function showDashboardPanel() {
console.log('showDashboardPanel called');
hideAllPanels();
const panel = document.getElementById('panel-dashboard');
if (panel) {
panel.style.display = 'flex';
loadDashboardModule();
console.log('Dashboard panel shown');
} else {
console.error('panel-dashboard not found');
}
}
// Open Klausur-Service (External Microservice)
function openKlausurService() {
// Pass auth token to klausur-service
const token = localStorage.getItem('auth_token');
const url = window.location.port === '8000'
? 'http://localhost:8086'
: window.location.origin.replace(':8000', ':8086');
// Open in new tab
const klausurWindow = window.open(url, '_blank');
// Try to pass token via postMessage after window loads
if (klausurWindow && token) {
setTimeout(() => {
try {
klausurWindow.postMessage({ type: 'AUTH_TOKEN', token: token }, url);
} catch (e) {
console.log('Could not pass token to klausur-service:', e);
}
}, 1000);
}
addActivity('🎓', 'Klausur-Service geoeffnet');
}
// =============================================
// AI PROMPT FUNCTIONS
// =============================================
let aiPromptAbortController = null;
function handleAiPromptKeydown(event) {
if (event.key === 'Enter' && !event.shiftKey) {
event.preventDefault();
sendAiPrompt();
}
}
function autoResizeTextarea(textarea) {
textarea.style.height = 'auto';
textarea.style.height = Math.min(textarea.scrollHeight, 120) + 'px';
}
async function sendAiPrompt() {
const input = document.getElementById('ai-prompt-input');
const sendBtn = document.getElementById('ai-prompt-send');
const responseDiv = document.getElementById('ai-response');
const responseText = document.getElementById('ai-response-text');
const responseModel = document.getElementById('ai-response-model');
const modelSelect = document.getElementById('ai-model-select');
const prompt = input?.value?.trim();
if (!prompt) return;
const model = modelSelect?.value || 'llama3.2:latest';
// Show loading state
sendBtn.disabled = true;
sendBtn.classList.add('loading');
sendBtn.textContent = '';
responseDiv.classList.add('active');
responseText.textContent = 'Denke nach...';
responseModel.textContent = model;
// Cancel previous request if exists
if (aiPromptAbortController) {
aiPromptAbortController.abort();
}
aiPromptAbortController = new AbortController();
try {
// Determine Ollama endpoint based on current host
let ollamaUrl = 'http://localhost:11434/api/generate';
if (window.location.hostname === 'macmini') {
ollamaUrl = 'http://macmini:11434/api/generate';
}
const response = await fetch(ollamaUrl, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
model: model,
prompt: prompt,
stream: true
}),
signal: aiPromptAbortController.signal
});
if (!response.ok) {
throw new Error(`Ollama error: ${response.status}`);
}
// Stream the response
const reader = response.body.getReader();
const decoder = new TextDecoder();
let fullResponse = '';
while (true) {
const { done, value } = await reader.read();
if (done) break;
const chunk = decoder.decode(value);
const lines = chunk.split('\\n').filter(l => l.trim());
for (const line of lines) {
try {
const data = JSON.parse(line);
if (data.response) {
fullResponse += data.response;
responseText.textContent = fullResponse;
}
} catch (e) {
// Ignore JSON parse errors for partial chunks
}
}
}
// Format the response
responseText.innerHTML = formatAiResponse(fullResponse);
// Log activity
addActivity('🤖', 'KI-Anfrage: ' + prompt.substring(0, 50) + (prompt.length > 50 ? '...' : ''));
} catch (error) {
if (error.name === 'AbortError') {
responseText.textContent = 'Anfrage abgebrochen.';
} else {
console.error('AI Prompt error:', error);
responseText.textContent = '❌ Fehler: ' + error.message + '\\n\\nBitte prüfen Sie, ob Ollama läuft (http://localhost:11434)';
}
} finally {
sendBtn.disabled = false;
sendBtn.classList.remove('loading');
sendBtn.textContent = '';
aiPromptAbortController = null;
}
}
function formatAiResponse(text) {
// Basic markdown-like formatting
let formatted = text
// Escape HTML
.replace(/&/g, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
// Code blocks
.replace(/```(\\w+)?\\n([\\s\\S]*?)```/g, '<pre><code>$2</code></pre>')
// Inline code
.replace(/`([^`]+)`/g, '<code>$1</code>')
// Bold
.replace(/\\*\\*([^*]+)\\*\\*/g, '<strong>$1</strong>')
// Italic
.replace(/\\*([^*]+)\\*/g, '<em>$1</em>')
// Line breaks
.replace(/\\n/g, '<br>');
return formatted;
}
// Load available models from Ollama
async function loadOllamaModels() {
const modelSelect = document.getElementById('ai-model-select');
if (!modelSelect) return;
try {
let ollamaUrl = 'http://localhost:11434/api/tags';
if (window.location.hostname === 'macmini') {
ollamaUrl = 'http://macmini:11434/api/tags';
}
const response = await fetch(ollamaUrl);
if (!response.ok) return;
const data = await response.json();
if (data.models && data.models.length > 0) {
modelSelect.innerHTML = data.models.map(m =>
`<option value="${m.name}">${m.name}</option>`
).join('');
}
} catch (error) {
console.log('Could not load Ollama models:', error.message);
}
}
// Initialize AI prompt on dashboard load
const originalLoadDashboardModule = loadDashboardModule;
loadDashboardModule = function() {
originalLoadDashboardModule();
loadOllamaModels();
};
"""