v2.4: Medienbibliothek+WebP, Varianten-Matrix, Litestream-Backups, intelligentere Analytics

P1 Medien: eigener Admin-Bereich /admin/medien (Grid, Mehrfach-Upload, Drag&Drop, Alt-Text, URL kopieren, Loeschen). Upload konvertiert JPG/PNG via sharp zu WebP (Qualitaet 82, max 2000px), Original wird verworfen; WebP/SVG/GIF/AVIF unveraendert; Konvertierungsfehler -> Original behalten statt 500. media um alt/width/height erweitert. Wiederverwendbarer Medien-Picker (public/media-picker.js) ersetzt den URL-Prompt im Block-Editor, Produkt-Editor (Karte/Galerie/Varianten-Bild), Slides und Popups. JSON-Quelle /api/admin/media (session-gesichert).

P2 Varianten: products.options_json + Tabelle product_variants. Produkt-Editor mit Options-Definition + Matrix-Generator (Preis-Override/Bestand/SKU/Bild/aktiv je Variante). PDP-Selektoren -> Variante; Cart/Checkout tragen sku+Options, Order-Item bekommt sku/variant, Variantenpreis serverseitig verifiziert. Produkte ohne Optionen unveraendert.

P3 Litestream: Binary im Dockerfile, docker-entrypoint.sh (Restore+replicate nur bei LITESTREAM_REPLICA_URL, sonst reiner Node-Start), litestream.yml, Backup-Status unter Einstellungen, README + .env.example.

P4 Analytics: Bestseller, Top-Suchbegriffe, Umsatz/Quelle, Umsatz-Zeitreihe, AOV, Wiederkaufrate, Lager-Warnungen.

Neue Dep sharp. +19 Unit-Tests (49 gesamt gruen), Build + Smoke (P1-P4) gruen.
This commit is contained in:
2026-06-18 08:09:57 +00:00
parent 30c41c355e
commit 50dfca59e1
28 changed files with 1147 additions and 66 deletions
+55
View File
@@ -241,3 +241,58 @@
/* v2.3 — Bewertungs-Zähler in der Nav */
.s-nav-badge{margin-left:auto;min-width:18px;height:18px;padding:0 5px;border-radius:999px;background:var(--accent);color:#fff;font-size:11px;font-weight:700;display:inline-flex;align-items:center;justify-content:center;line-height:1}
/* v2.4 — Medien-Picker-Modal + Varianten-Matrix */
.mp-overlay{position:fixed;inset:0;background:rgba(20,24,30,.46);backdrop-filter:blur(3px);display:none;align-items:center;justify-content:center;z-index:300;padding:24px}
.mp-overlay.open{display:flex;animation:fade .15s var(--s-ease)}
.mp-modal{width:min(920px,96vw);max-height:90vh;display:flex;flex-direction:column;background:var(--s-bg);border:1px solid var(--s-border);border-radius:16px;box-shadow:0 24px 70px rgba(0,0,0,.32);overflow:hidden;animation:pop .18s var(--s-ease)}
.mp-head{display:flex;align-items:center;justify-content:space-between;padding:14px 18px;border-bottom:1px solid var(--s-border);font-family:var(--s-display);font-size:16px;color:var(--s-ink)}
.mp-head-actions{display:flex;align-items:center;gap:8px}
.mp-upbtn{display:inline-flex;align-items:center;gap:6px;padding:7px 13px;border:1px solid var(--accent);color:var(--accent-dark);border-radius:9px;font-size:13px;font-weight:600;cursor:pointer;background:var(--s-surface)}
.mp-upbtn:hover{background:var(--s-acc-ring)}
.mp-close{width:30px;height:30px;border:none;background:var(--s-sunken);border-radius:8px;cursor:pointer;color:var(--s-subtle);font-size:15px}
.mp-close:hover{background:var(--s-border);color:var(--s-ink)}
.mp-drop{margin:14px 18px 0;padding:12px;border:1px dashed var(--s-border-2);border-radius:10px;text-align:center;font-size:12.5px;color:var(--s-faint);transition:.15s}
.mp-drop.over{border-color:var(--accent);background:var(--s-acc-ring);color:var(--accent-dark)}
.mp-msg{margin:8px 18px 0;font-size:12.5px;min-height:0;color:var(--s-faint);display:none}
.mp-msg.show{display:block}
.mp-msg.ok{color:var(--s-green-t,#1a7f4b)}
.mp-msg.err{color:var(--s-red-t,#b3261e)}
.mp-grid{flex:1;overflow:auto;display:grid;grid-template-columns:repeat(auto-fill,minmax(140px,1fr));gap:13px;padding:16px 18px}
.mp-empty{grid-column:1/-1;text-align:center;color:var(--s-faint);padding:40px 10px;font-size:13.5px}
.mp-card{position:relative;border:1px solid var(--s-border);border-radius:11px;overflow:hidden;background:var(--s-surface);cursor:pointer;transition:.15s}
.mp-card:hover{box-shadow:var(--s-shadow-pop);transform:translateY(-2px)}
.mp-card.sel{border-color:var(--accent);box-shadow:0 0 0 2px var(--s-acc-ring)}
.mp-thumb{height:104px;background:var(--s-sunken)}
.mp-thumb img{width:100%;height:100%;object-fit:cover}
.mp-meta{padding:7px 9px;display:flex;flex-direction:column;gap:2px}
.mp-name{font-size:11px;color:var(--s-ink);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.mp-size{font-size:10.5px;color:var(--s-faint)}
.mp-card-acts{position:absolute;top:6px;right:6px;display:flex;gap:4px;opacity:0;transition:.15s}
.mp-card:hover .mp-card-acts{opacity:1}
.mp-ico{width:26px;height:26px;border:none;border-radius:7px;background:rgba(255,255,255,.92);box-shadow:var(--s-shadow);cursor:pointer;font-size:12px;display:grid;place-items:center}
.mp-ico:hover{background:#fff}
.mp-del:hover{color:#b3261e}
.mp-foot{display:flex;align-items:center;gap:10px;padding:12px 18px;border-top:1px solid var(--s-border)}
.mp-sel{font-size:12.5px;color:var(--s-subtle);font-weight:600}
.mp-btn{padding:8px 16px;border:1px solid var(--s-border);border-radius:9px;background:var(--s-surface);font-size:13px;font-weight:600;cursor:pointer;color:var(--s-ink);font-family:inherit}
.mp-btn:hover{border-color:var(--s-border-2)}
.mp-primary{background:var(--accent);border-color:var(--accent);color:#fff}
.mp-primary:hover{background:var(--accent-dark);border-color:var(--accent-dark)}
/* Bild-Feld mit Vorschau + Picker */
.s-imgfield{display:flex;gap:8px;align-items:center}
.s-imgfield .s-input{flex:1}
.s-imgthumb{width:42px;height:42px;border-radius:8px;object-fit:cover;background:var(--s-sunken);border:1px solid var(--s-border);flex:none}
/* Varianten-Matrix */
.s-vopts{display:flex;flex-direction:column;gap:10px}
.s-vopt-row{display:grid;grid-template-columns:160px 1fr auto;gap:8px;align-items:center}
.s-vtable{width:100%;border-collapse:collapse;font-size:12.5px}
.s-vtable th,.s-vtable td{padding:7px 8px;border-bottom:1px solid var(--s-line-soft);text-align:left;vertical-align:middle}
.s-vtable th{font-size:11px;text-transform:uppercase;letter-spacing:.05em;color:var(--s-faint);font-weight:700}
.s-vtable input.s-vin{width:100%;padding:6px 8px;border:1px solid var(--s-border);border-radius:7px;font-size:12.5px;font-family:inherit}
.s-vtable td.num input{text-align:right}
.s-vtable .s-vimg{width:34px;height:34px;border-radius:6px;object-fit:cover;background:var(--s-sunken);border:1px solid var(--s-border);cursor:pointer}
.s-vrow-off{opacity:.5}
.s-vbadge{font-size:11px;color:var(--s-subtle)}