v2: Session-Login & Rollen, Premium-Admin, Visual-Block-Builder, KI-/MCP-API
- Auth-Umbau: Session-Login (signiertes HMAC-Cookie, scrypt-Hashing) statt Basic-Auth; users-/audit-Tabellen, Initial-Owner aus ENV, Rate-Limit, konfigurierbarer ADMIN_PATH (Middleware-Rewrite), Rollen-Gate (owner/redaktion/versand), Nutzerverwaltung, Audit-Log, Login/Logout/Konto-Seiten. - Premium-Pass: Command-Palette (Cmd-K), Toasts, Account-Menue, aufgewertetes Dashboard (KPI-Trend+Sparkline, Aktivitaets-Feed, Schnellaktionen), schoene Empty-States. - Block-Builder: pages.blocks, Vollbild-Editor (Liste/Live-Vorschau/Settings, Desktop/Mobil), 10 Block-Typen, Storefront-BlockRenderer auf /seite/[slug], Save-Endpoint. - KI-Editierbarkeit: token-gesicherte /api/admin/* (CRUD), Manifest /api/admin + /ai-admin.txt, MCP-Server unter mcp/ (14 Tools). - Docs: README + .env.example + mcp/README aktualisiert.
This commit is contained in:
@@ -165,3 +165,75 @@
|
||||
|
||||
@media(prefers-reduced-motion:reduce){*{transition:none!important;animation:none!important}}
|
||||
@media(max-width:860px){.admin-shell{grid-template-columns:1fr}.s-side{position:static;height:auto}.s-nav{flex-direction:row;flex-wrap:wrap}.s-nav a.active::before{display:none}.s-kpis{grid-template-columns:1fr 1fr}.s-form-grid{grid-template-columns:1fr}.s-two-col{grid-template-columns:1fr}}
|
||||
|
||||
/* ===== v2: Login ===== */
|
||||
.login-body{display:flex;min-height:100vh;align-items:center;justify-content:center;background:
|
||||
radial-gradient(1200px 600px at 50% -10%, color-mix(in srgb,var(--accent) 9%, transparent), transparent 60%),
|
||||
var(--s-bg)}
|
||||
.login-wrap{width:100%;max-width:420px;padding:24px}
|
||||
.login-card{background:var(--s-surface);border:1px solid var(--s-border);border-radius:18px;padding:34px 32px 26px;box-shadow:var(--s-shadow-pop);text-align:left}
|
||||
.login-logo{width:52px;height:52px;border-radius:14px;background:var(--accent);color:#fff;display:grid;place-items:center;font-weight:800;font-size:23px;margin:0 auto 16px;box-shadow:inset 0 1px 0 rgba(255,255,255,.25),0 6px 18px -8px color-mix(in srgb,var(--accent) 70%, transparent)}
|
||||
.login-title{font-family:var(--s-display);font-weight:560;font-size:25px;color:var(--s-ink);text-align:center;margin:0;letter-spacing:-.02em}
|
||||
.login-sub{text-align:center;color:var(--s-subtle);font-size:13.5px;margin:4px 0 22px}
|
||||
.login-error{background:var(--s-red);color:var(--s-red-t);border:1px solid color-mix(in srgb,var(--s-red-t) 26%, #fff);padding:10px 14px;border-radius:10px;font-weight:600;font-size:13px;margin-bottom:16px;display:flex;align-items:center;gap:8px}
|
||||
.login-error::before{content:'!';display:inline-grid;place-items:center;width:18px;height:18px;border-radius:50%;background:var(--s-red-t);color:#fff;font-size:12px;font-weight:800;flex:none}
|
||||
.login-remember{margin:2px 0 18px;font-size:13.5px;color:var(--s-text)}
|
||||
.login-submit{width:100%;justify-content:center;padding:11px;font-size:14px}
|
||||
.login-foot{text-align:center;color:var(--s-faint);font-size:11.5px;margin:18px 0 0}
|
||||
|
||||
/* ===== v2: Account-Menü + ⌘K-Trigger ===== */
|
||||
.s-cmdk-trigger{gap:9px;padding:7px 10px}
|
||||
.s-kbd{font-size:11px;font-weight:700;color:var(--s-subtle);background:var(--s-sunken);border:1px solid var(--s-border);border-radius:6px;padding:1px 6px;letter-spacing:.02em}
|
||||
.s-account{position:relative}
|
||||
.s-account-btn{display:flex;align-items:center;gap:9px;background:var(--s-surface);border:1px solid var(--s-border-2);border-radius:var(--s-radius-sm);padding:5px 11px 5px 6px;cursor:pointer;font-family:inherit;transition:background .15s,box-shadow .15s,transform .15s}
|
||||
.s-account-btn:hover{background:var(--s-bg);box-shadow:var(--s-shadow);transform:translateY(-1px)}
|
||||
.s-acct-av{width:28px;height:28px;border-radius:8px;background:var(--accent);color:#fff;display:grid;place-items:center;font-weight:800;font-size:13px;flex:none}
|
||||
.s-acct-meta{display:flex;flex-direction:column;line-height:1.15;text-align:left}
|
||||
.s-acct-name{font-size:13px;font-weight:600;color:var(--s-ink);max-width:130px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
|
||||
.s-acct-role{font-size:11px;color:var(--s-faint)}
|
||||
.s-account-menu{position:absolute;right:0;top:calc(100% + 8px);background:var(--s-surface);border:1px solid var(--s-border);border-radius:12px;box-shadow:var(--s-shadow-pop);padding:6px;min-width:170px;z-index:40;display:flex;flex-direction:column;gap:2px}
|
||||
.s-account-menu a{padding:9px 12px;border-radius:8px;font-size:13.5px;font-weight:500;color:var(--s-text);transition:background .12s}
|
||||
.s-account-menu a:hover{background:var(--s-bg);color:var(--s-ink)}
|
||||
.s-account-menu a.danger{color:var(--s-red-t)}
|
||||
.s-account-menu a.danger:hover{background:var(--s-red)}
|
||||
|
||||
/* ===== v2: Command-Palette ===== */
|
||||
.s-cmdk{position:fixed;inset:0;z-index:100;display:flex;align-items:flex-start;justify-content:center;padding-top:14vh}
|
||||
.s-cmdk[hidden]{display:none}
|
||||
.s-cmdk-backdrop{position:absolute;inset:0;background:rgba(43,38,32,.34);backdrop-filter:blur(2px);animation:fade .15s var(--s-ease)}
|
||||
.s-cmdk-panel{position:relative;width:100%;max-width:560px;background:var(--s-surface);border:1px solid var(--s-border);border-radius:16px;box-shadow:var(--s-shadow-pop);overflow:hidden;animation:pop .16s var(--s-ease)}
|
||||
.s-cmdk-input{width:100%;border:none;border-bottom:1px solid var(--s-line-soft);padding:17px 20px;font:inherit;font-size:16px;color:var(--s-ink);background:transparent;outline:none}
|
||||
.s-cmdk-input::placeholder{color:var(--s-faint)}
|
||||
.s-cmdk-list{list-style:none;margin:0;padding:8px;max-height:46vh;overflow:auto}
|
||||
.s-cmdk-item{display:flex;align-items:center;justify-content:space-between;padding:11px 14px;border-radius:10px;cursor:pointer;font-size:14px;color:var(--s-text)}
|
||||
.s-cmdk-item em{font-style:normal;font-size:11px;color:var(--s-faint);text-transform:uppercase;letter-spacing:.05em;font-weight:700}
|
||||
.s-cmdk-item.active{background:var(--s-acc-l);color:var(--accent-dark)}
|
||||
.s-cmdk-item.active em{color:color-mix(in srgb,var(--accent-dark) 70%, transparent)}
|
||||
|
||||
/* ===== v2: Toasts ===== */
|
||||
.s-toasts{position:fixed;right:20px;bottom:20px;z-index:120;display:flex;flex-direction:column;gap:10px;align-items:flex-end}
|
||||
.s-toast{background:var(--s-ink);color:#fff;padding:12px 18px;border-radius:11px;font-size:13.5px;font-weight:600;box-shadow:var(--s-shadow-pop);opacity:0;transform:translateY(8px);transition:opacity .25s var(--s-ease),transform .25s var(--s-ease);max-width:340px}
|
||||
.s-toast.show{opacity:1;transform:translateY(0)}
|
||||
.s-toast.ok{background:#2f6b4f}
|
||||
.s-toast.err{background:var(--s-red-t)}
|
||||
|
||||
/* ===== v2: Empty-States, Skeleton, KPI-Trend ===== */
|
||||
.s-emptystate{text-align:center;padding:54px 24px;display:flex;flex-direction:column;align-items:center;gap:12px}
|
||||
.s-emptystate .es-icon{width:56px;height:56px;border-radius:16px;background:var(--s-acc-l);color:var(--accent-dark);display:grid;place-items:center}
|
||||
.s-emptystate .es-icon svg{width:28px;height:28px}
|
||||
.s-emptystate h3{font-family:var(--s-display);font-weight:560;font-size:18px;color:var(--s-ink);margin:0}
|
||||
.s-emptystate p{color:var(--s-subtle);font-size:13.5px;margin:0;max-width:360px}
|
||||
.s-kpi-trend{display:inline-flex;align-items:center;gap:4px;font-size:12px;font-weight:700;margin-top:6px;font-variant-numeric:tabular-nums}
|
||||
.s-kpi-trend.up{color:var(--s-green-t)}.s-kpi-trend.down{color:var(--s-red-t)}
|
||||
.s-kpi-spark{height:30px;margin-top:8px}
|
||||
.s-quick{display:grid;grid-template-columns:repeat(auto-fit,minmax(150px,1fr));gap:12px}
|
||||
.s-quick a{display:flex;align-items:center;gap:11px;padding:14px 16px;border:1px solid var(--s-border);border-radius:12px;background:var(--s-surface);box-shadow:var(--s-shadow);font-weight:600;color:var(--s-ink);transition:transform .15s var(--s-ease),box-shadow .15s}
|
||||
.s-quick a:hover{transform:translateY(-2px);box-shadow:var(--s-shadow-pop)}
|
||||
.s-quick a svg{width:20px;height:20px;color:var(--accent)}
|
||||
.s-feed{display:flex;flex-direction:column}
|
||||
.s-feed-row{display:flex;align-items:center;gap:12px;padding:11px 22px;border-bottom:1px solid var(--s-line-soft);font-size:13px}
|
||||
.s-feed-row:last-child{border-bottom:none}
|
||||
.s-feed-dot{width:8px;height:8px;border-radius:50%;background:var(--accent);flex:none}
|
||||
.s-feed-row .t{color:var(--s-faint);margin-left:auto;font-size:12px;white-space:nowrap}
|
||||
@keyframes fade{from{opacity:0}to{opacity:1}}
|
||||
@keyframes pop{from{opacity:0;transform:translateY(-6px) scale(.98)}to{opacity:1;transform:none}}
|
||||
|
||||
Reference in New Issue
Block a user