Drei-dimensionale Aggregations-Sicht über Bundesland × Partei × Wahlperiode mit minimalem Frontend. Backend (`app/auswertungen.py`): - `aggregate_matrix(filter_wp=None)` — 2D-Matrix Bundesland × Partei mit (n, Ø-Score) pro Zelle, optional gefiltert nach Wahlperiode - `aggregate_zeitreihe(bundesland, partei)` — Score-Verlauf einer (BL, Partei)-Kombination über alle bekannten WPs - `export_long_format()` — Long-Format-CSV-Export für externe Tools (deckt #45 vollständig ab) - Partei-Auflösung läuft strikt durch `normalize_partei()` aus #55 — damit wird BB-`FREIE WÄHLER` korrekt als `BVB-FW` aggregiert und NICHT mit dem RP-FW zusammengezählt Wahlperioden-Helper (`app/wahlperioden.py`): - `wahlperiode_for(datum, bundesland)` mappt ein ISO-Datum + BL auf eine Kennung wie `"NRW-WP18"` oder `"MV-WP7"` (Vorgänger-WP). Single Source of Truth ist `BUNDESLAENDER[bl].wahlperiode_start` - `all_wahlperioden()` für UI-Filter-Dropdowns Endpoints in `app/main.py`: - `GET /auswertungen` — HTML-Seite (neues Template) - `GET /api/auswertungen/matrix?wahlperiode=NRW-WP18` — JSON-Matrix - `GET /api/auswertungen/zeitreihe?bundesland=MV&partei=CDU` — JSON-Verlauf - `GET /api/auswertungen/export.csv` — CSV-Download Frontend (`app/templates/auswertungen.html`): - Statisches Template mit Vanilla-JS, kein Build-Step - Wahlperioden-Dropdown + Reload-Button + CSV-Export-Button - Matrix-Tabelle mit Score-Color-Coding (rot ≤ 3, gelb 3-6, grün > 6) - Sticky-Bundesland-Spalte für horizontales Scrolling Tests (`tests/test_auswertungen.py`): - 19 Cases mit in-memory SQLite-Fixture - Verifiziert WP-Mapping, Matrix-Aggregation, Koalitions-Counting, WP-Filter-Korrektheit, BVB-FW-Disambiguierung in der Matrix, CSV-Long-Format - 176 Unit-Tests grün (157 alt + 19 neu) Refs: #58, #45, #59 (Phase C) Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
202 lines
7.2 KiB
HTML
202 lines
7.2 KiB
HTML
<!DOCTYPE html>
|
||
<html lang="de">
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||
<title>Auswertungen — {{ app_name }}</title>
|
||
<style>
|
||
:root {
|
||
--color-darkgray: #5a5a5a;
|
||
--color-green: #889e33;
|
||
--color-blue: #009da5;
|
||
--color-lightgray: #bfbfbf;
|
||
--color-bg: #f5f5f5;
|
||
--color-orange: #F7941D;
|
||
--color-red: #d00000;
|
||
}
|
||
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||
body {
|
||
font-family: 'Avenir', 'Segoe UI', sans-serif;
|
||
color: var(--color-darkgray);
|
||
line-height: 1.6;
|
||
background: var(--color-bg);
|
||
}
|
||
.header {
|
||
background: white;
|
||
padding: 1rem 2rem;
|
||
border-bottom: 1px solid var(--color-lightgray);
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
flex-wrap: wrap;
|
||
gap: 1rem;
|
||
}
|
||
.header h1 { color: var(--color-blue); font-size: 1.3rem; }
|
||
.header nav a {
|
||
color: var(--color-blue);
|
||
text-decoration: none;
|
||
margin-right: 1rem;
|
||
font-size: 0.9rem;
|
||
}
|
||
.header nav a:hover { text-decoration: underline; }
|
||
main { max-width: 1400px; margin: 1.5rem auto; padding: 0 2rem; }
|
||
.controls {
|
||
background: white;
|
||
padding: 1rem;
|
||
border-radius: 4px;
|
||
margin-bottom: 1rem;
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 1rem;
|
||
flex-wrap: wrap;
|
||
}
|
||
.controls label { font-size: 0.9rem; }
|
||
.controls select, .controls button {
|
||
padding: 0.4rem 0.7rem;
|
||
border: 1px solid var(--color-lightgray);
|
||
border-radius: 3px;
|
||
font-size: 0.9rem;
|
||
background: white;
|
||
color: var(--color-darkgray);
|
||
cursor: pointer;
|
||
}
|
||
.controls button.export {
|
||
background: var(--color-blue);
|
||
color: white;
|
||
border-color: var(--color-blue);
|
||
}
|
||
.matrix-wrap { background: white; padding: 1rem; border-radius: 4px; overflow-x: auto; }
|
||
table.matrix {
|
||
border-collapse: collapse;
|
||
width: 100%;
|
||
font-size: 0.85rem;
|
||
}
|
||
table.matrix th, table.matrix td {
|
||
border: 1px solid var(--color-lightgray);
|
||
padding: 0.4rem 0.6rem;
|
||
text-align: center;
|
||
}
|
||
table.matrix th {
|
||
background: var(--color-bg);
|
||
font-weight: 600;
|
||
color: var(--color-darkgray);
|
||
}
|
||
table.matrix th.row-header {
|
||
background: var(--color-blue);
|
||
color: white;
|
||
text-align: left;
|
||
position: sticky;
|
||
left: 0;
|
||
}
|
||
table.matrix .empty { color: var(--color-lightgray); }
|
||
table.matrix .score-high { background: rgba(136, 158, 51, 0.25); font-weight: 600; }
|
||
table.matrix .score-mid { background: rgba(247, 148, 29, 0.18); }
|
||
table.matrix .score-low { background: rgba(208, 0, 0, 0.18); font-weight: 600; }
|
||
.meta {
|
||
font-size: 0.8rem;
|
||
color: var(--color-lightgray);
|
||
margin-top: 0.6rem;
|
||
}
|
||
.empty-state {
|
||
background: white;
|
||
padding: 2rem;
|
||
text-align: center;
|
||
border-radius: 4px;
|
||
color: var(--color-lightgray);
|
||
}
|
||
</style>
|
||
</head>
|
||
<body>
|
||
<div class="header">
|
||
<h1>Auswertungen — Bundesland × Partei × Wahlperiode</h1>
|
||
<nav>
|
||
<a href="/">← zurück zur Suche</a>
|
||
<a href="/quellen">Quellen</a>
|
||
</nav>
|
||
</div>
|
||
|
||
<main>
|
||
<div class="controls">
|
||
<label>Wahlperiode:
|
||
<select id="wp-filter">
|
||
<option value="">— alle WPs —</option>
|
||
{% for wp in wahlperioden %}
|
||
<option value="{{ wp }}">{{ wp }}</option>
|
||
{% endfor %}
|
||
</select>
|
||
</label>
|
||
<button id="reload">Anwenden</button>
|
||
<button class="export" id="export-csv">CSV-Export (alle Daten)</button>
|
||
</div>
|
||
|
||
<div id="matrix-container" class="matrix-wrap">
|
||
<div class="empty-state">Lade Matrix …</div>
|
||
</div>
|
||
|
||
<div class="meta" id="meta"></div>
|
||
</main>
|
||
|
||
<script>
|
||
const wpFilter = document.getElementById('wp-filter');
|
||
const reloadBtn = document.getElementById('reload');
|
||
const exportBtn = document.getElementById('export-csv');
|
||
const container = document.getElementById('matrix-container');
|
||
const meta = document.getElementById('meta');
|
||
|
||
function scoreClass(avg) {
|
||
if (avg === null || avg === undefined) return '';
|
||
if (avg >= 6) return 'score-high';
|
||
if (avg >= 3) return 'score-mid';
|
||
return 'score-low';
|
||
}
|
||
|
||
async function loadMatrix() {
|
||
const wp = wpFilter.value;
|
||
const url = wp
|
||
? `/api/auswertungen/matrix?wahlperiode=${encodeURIComponent(wp)}`
|
||
: '/api/auswertungen/matrix';
|
||
container.innerHTML = '<div class="empty-state">Lade Matrix …</div>';
|
||
try {
|
||
const r = await fetch(url);
|
||
const data = await r.json();
|
||
if (!data.bundeslaender.length) {
|
||
container.innerHTML = '<div class="empty-state">Keine Assessments für diesen Filter.</div>';
|
||
meta.textContent = '';
|
||
return;
|
||
}
|
||
let html = '<table class="matrix"><thead><tr><th class="row-header">Bundesland</th>';
|
||
for (const partei of data.parteien) {
|
||
html += `<th>${partei}</th>`;
|
||
}
|
||
html += '</tr></thead><tbody>';
|
||
for (const bl of data.bundeslaender) {
|
||
html += `<tr><th class="row-header">${bl}</th>`;
|
||
for (const partei of data.parteien) {
|
||
const cell = (data.cells[bl] || {})[partei];
|
||
if (cell) {
|
||
html += `<td class="${scoreClass(cell.avg)}">${cell.avg.toFixed(1)}<br><small>n=${cell.n}</small></td>`;
|
||
} else {
|
||
html += '<td class="empty">—</td>';
|
||
}
|
||
}
|
||
html += '</tr>';
|
||
}
|
||
html += '</tbody></table>';
|
||
container.innerHTML = html;
|
||
meta.textContent = `${data.total} Assessment(s) | Filter: ${data.filter_wp || 'alle WPs'}`;
|
||
} catch (e) {
|
||
container.innerHTML = `<div class="empty-state">Fehler beim Laden: ${e}</div>`;
|
||
}
|
||
}
|
||
|
||
reloadBtn.addEventListener('click', loadMatrix);
|
||
wpFilter.addEventListener('change', loadMatrix);
|
||
exportBtn.addEventListener('click', () => {
|
||
window.location.href = '/api/auswertungen/export.csv';
|
||
});
|
||
|
||
loadMatrix();
|
||
</script>
|
||
</body>
|
||
</html>
|