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>
955 lines
24 KiB
Python
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, '&')
|
|
.replace(/</g, '<')
|
|
.replace(/>/g, '>')
|
|
// 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();
|
|
};
|
|
"""
|