@@ -130,6 +519,224 @@
}
.btn-nav:hover { background: #0B1023; color: #fff; }
+ /* MOCKUPS */
+ .mockups {
+ padding: 5rem 2rem;
+ background: #0B1023;
+ color: #fff;
+ overflow: hidden;
+ }
+ .mockups .eyebrow-dark {
+ font-size: 0.8rem;
+ font-weight: 600;
+ text-transform: uppercase;
+ letter-spacing: 1.5px;
+ color: #F97316;
+ text-align: center;
+ margin-bottom: 1rem;
+ }
+ .mockups h2 { color: #fff; text-align: center; margin-bottom: 3rem; }
+
+ .devices {
+ display: flex;
+ align-items: flex-start;
+ justify-content: center;
+ gap: 3rem;
+ flex-wrap: wrap;
+ }
+
+ /* Browser Mockup */
+ .device-browser {
+ flex: 1;
+ min-width: 320px;
+ max-width: 580px;
+ background: #1a2035;
+ border-radius: 12px;
+ overflow: hidden;
+ box-shadow: 0 24px 60px rgba(0,0,0,0.5);
+ border: 1px solid #2a3347;
+ }
+ .browser-bar {
+ background: #252d42;
+ padding: 0.6rem 1rem;
+ display: flex;
+ align-items: center;
+ gap: 0.75rem;
+ }
+ .browser-dots { display: flex; gap: 5px; }
+ .browser-dots span {
+ width: 10px; height: 10px;
+ border-radius: 50%;
+ background: #3a4055;
+ }
+ .browser-url {
+ flex: 1;
+ background: #1a2035;
+ border-radius: 4px;
+ padding: 0.2rem 0.6rem;
+ font-size: 0.7rem;
+ color: #6b7a9a;
+ font-family: monospace;
+ }
+ .browser-screen { display: flex; height: 420px; overflow: hidden; }
+
+ /* CSS Dashboard */
+ .dash-sidebar {
+ width: 120px;
+ background: #0B1023;
+ padding: 1rem 0.75rem;
+ flex-shrink: 0;
+ display: flex;
+ flex-direction: column;
+ gap: 0.25rem;
+ }
+ .dash-brand-logo { width: 100%; margin-bottom: 1rem; }
+ .dash-nav-item.schmidt { background: rgba(42,155,87,0.15); color: #2A9B57; }
+ .dash-nav-item {
+ padding: 0.4rem 0.5rem;
+ border-radius: 5px;
+ font-size: 0.65rem;
+ color: #6b7a9a;
+ background: none;
+ border: none;
+ cursor: pointer;
+ text-align: left;
+ width: 100%;
+ transition: all 0.15s;
+ }
+ .dash-nav-item:hover { color: #fff; background: rgba(255,255,255,0.06); }
+ .dash-nav-item.active { background: rgba(249,115,22,0.15); color: #F97316; }
+
+ .dash-main {
+ flex: 1;
+ padding: 1rem;
+ background: #F5F7FA;
+ overflow: hidden;
+ }
+ .dash-title { font-size: 0.9rem; font-weight: 800; color: #0B1023; }
+ .dash-date { font-size: 0.65rem; color: #888; margin-bottom: 0.75rem; }
+
+ .dash-stats {
+ display: grid;
+ grid-template-columns: repeat(4, 1fr);
+ gap: 0.4rem;
+ margin-bottom: 0.75rem;
+ }
+ .dash-stat {
+ background: #fff;
+ border-radius: 6px;
+ padding: 0.4rem;
+ border-left: 2.5px solid #ddd;
+ }
+ .dash-stat.ok { border-color: #16a34a; }
+ .dash-stat.warn { border-color: #ea580c; }
+ .dash-stat.crit { border-color: #dc2626; }
+ .dash-num { font-size: 1rem; font-weight: 800; color: #0B1023; line-height: 1; }
+ .dash-label { font-size: 0.5rem; color: #888; margin-top: 0.15rem; }
+
+ .dash-refresh { font-size: 0.55rem; color: #aaa; }
+ .dash-entries-title { font-size: 0.7rem; font-weight: 700; color: #0B1023; margin-bottom: 0.4rem; }
+ .dash-empty { font-size: 0.65rem; color: #aaa; padding: 0.5rem 0; }
+
+ /* Protokoll-Tabelle */
+ .proto-table { width: 100%; border-collapse: collapse; font-size: 0.72rem; margin-top: 0.5rem; }
+ .proto-table th { background: #F5F7FA; padding: 0.4rem 0.5rem; text-align: left; color: #666; font-size: 0.65rem; text-transform: uppercase; letter-spacing: 0.3px; }
+ .proto-table td { padding: 0.45rem 0.5rem; border-bottom: 1px solid #f0f0f0; color: #333; }
+ .proto-station { font-weight: 600; color: #0B1023; max-width: 110px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
+ .proto-time { color: #888; }
+
+ /* Stationen-Grid */
+ .stations-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 0.4rem; margin-top: 0.5rem; }
+ .station-card { background: #F5F7FA; border-radius: 6px; padding: 0.5rem; }
+ .station-card-icon { width: 16px; height: 16px; margin-bottom: 0.25rem; filter: invert(30%) sepia(20%) saturate(500%) hue-rotate(182deg); }
+ .station-card-name { font-size: 0.6rem; font-weight: 700; color: #0B1023; }
+ .station-card-temp { font-size: 0.55rem; color: #888; margin-top: 0.1rem; }
+ .dash-entry {
+ background: #fff;
+ border-radius: 6px;
+ padding: 0.4rem 0.5rem;
+ font-size: 0.6rem;
+ display: flex;
+ align-items: center;
+ gap: 0.4rem;
+ margin-bottom: 0.3rem;
+ }
+ .entry-badge {
+ padding: 0.1rem 0.35rem;
+ border-radius: 10px;
+ font-weight: 700;
+ font-size: 0.55rem;
+ white-space: nowrap;
+ }
+ .entry-station { flex: 1; font-weight: 600; color: #0B1023; }
+ .entry-temp { color: #555; }
+ .entry-who { color: #aaa; white-space: nowrap; }
+
+ /* Phone Mockup */
+ .device-phone {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ gap: 1rem;
+ }
+ .phone-frame {
+ width: 220px;
+ background: #1a1a2e;
+ border-radius: 36px;
+ padding: 12px;
+ box-shadow: 0 24px 60px rgba(0,0,0,0.6), inset 0 0 0 1px #2a2a4a;
+ }
+ .phone-notch {
+ width: 60px;
+ height: 20px;
+ background: #1a1a2e;
+ border-radius: 10px;
+ margin: 0 auto 8px;
+ border: 1.5px solid #2a2a4a;
+ }
+ .phone-screen {
+ background: #fff;
+ border-radius: 24px;
+ overflow: hidden;
+ height: 490px;
+ position: relative;
+ }
+ .phone-screen iframe {
+ position: absolute;
+ top: 0;
+ left: 0;
+ width: 390px;
+ height: 1200px;
+ border: none;
+ transform: scale(0.502);
+ transform-origin: top left;
+ pointer-events: auto;
+ }
+ .phone-home {
+ width: 60px;
+ height: 4px;
+ background: #3a3a5a;
+ border-radius: 2px;
+ margin: 10px auto 0;
+ }
+ .phone-caption {
+ font-size: 0.8rem;
+ color: #6b7a9a;
+ text-align: center;
+ }
+
+ @media (max-width: 768px) {
+ .devices {
+ flex-direction: column;
+ align-items: center;
+ }
+ .device-browser {
+ max-width: 100%;
+ min-width: unset;
+ }
+ .browser-screen { height: 220px; }
+ }
+
/* HERO */
.hero {
background: linear-gradient(135deg, #0B1023 0%, #16213e 100%);
@@ -254,6 +861,84 @@
.feature-card h3 { font-size: 1.1rem; margin-bottom: 0.5rem; }
.feature-card p { color: #555; font-size: 0.95rem; }
+ /* PRICING */
+ .pricing { padding: 5rem 2rem; background: #F5F7FA; }
+
+ .pricing-intro { text-align: center; margin-bottom: 3rem; }
+ .pricing-sub { color: #555; margin-top: 1rem; margin-bottom: 1.5rem; font-size: 1.05rem; max-width: 640px; margin-left: auto; margin-right: auto; }
+
+ .pricing-math {
+ display: inline-flex;
+ align-items: center;
+ gap: 0.75rem;
+ background: #fff;
+ border: 1.5px solid #F97316;
+ border-radius: 10px;
+ padding: 0.6rem 1.25rem;
+ font-size: 0.9rem;
+ margin-top: 0.5rem;
+ }
+ .math-example { color: #555; }
+ .math-arrow { color: #F97316; font-size: 1.2rem; }
+ .math-result { font-weight: 700; color: #0B1023; }
+
+ .plan-desc { font-size: 0.85rem; color: #888; margin-bottom: 1.25rem; margin-top: -0.75rem; }
+ .pricing-note { text-align: center; margin-top: 2rem; font-size: 0.85rem; color: #aaa; }
+
+ .pricing-grid {
+ display: grid;
+ grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
+ gap: 1.5rem;
+ align-items: stretch;
+ }
+
+ .plan {
+ background: #fff;
+ border-radius: 16px;
+ padding: 2rem;
+ border: 1.5px solid #eee;
+ display: flex;
+ flex-direction: column;
+ position: relative;
+ }
+ .plan-featured {
+ border-color: #F97316;
+ box-shadow: 0 4px 24px rgba(249,115,22,0.12);
+ }
+ .plan-badge {
+ position: absolute;
+ top: -12px;
+ left: 50%;
+ transform: translateX(-50%);
+ background: #F97316;
+ color: #fff;
+ padding: 0.2rem 0.9rem;
+ border-radius: 20px;
+ font-size: 0.75rem;
+ font-weight: 700;
+ }
+ .plan-name { font-size: 0.85rem; font-weight: 700; color: #888; text-transform: uppercase; letter-spacing: 1px; margin-bottom: 0.75rem; }
+ .plan-price { font-size: 2rem; font-weight: 800; color: #0B1023; margin-bottom: 1.5rem; }
+ .plan-price span { font-size: 1rem; font-weight: 400; color: #888; }
+
+ .plan-features { list-style: none; margin: 0 0 2rem; padding: 0; display: flex; flex-direction: column; gap: 0.6rem; flex: 1; }
+ .plan-features li { font-size: 0.9rem; color: #444; }
+
+ .btn-plan {
+ display: block;
+ text-align: center;
+ padding: 0.75rem;
+ border-radius: 8px;
+ font-weight: 700;
+ font-size: 0.95rem;
+ border: 1.5px solid #0B1023;
+ color: #0B1023;
+ transition: all 0.15s;
+ }
+ .btn-plan:hover { background: #0B1023; color: #fff; }
+ .btn-plan-featured { background: #F97316; border-color: #F97316; color: #fff; }
+ .btn-plan-featured:hover { background: #ea6c10; border-color: #ea6c10; }
+
/* CTA */
.cta {
background: #0B1023;
@@ -269,6 +954,44 @@
padding: 1.5rem 2rem;
border-top: 1px solid #f0f0f0;
}
+ /* MODAL */
+ .modal-overlay {
+ position: fixed; inset: 0;
+ background: rgba(11,16,35,0.7);
+ display: flex; align-items: center; justify-content: center;
+ z-index: 100; padding: 1rem;
+ backdrop-filter: blur(4px);
+ }
+ .modal {
+ background: #fff; border-radius: 16px;
+ padding: 2rem; width: 100%; max-width: 540px;
+ box-shadow: 0 24px 60px rgba(0,0,0,0.3);
+ }
+ .modal-header { display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1.5rem; }
+ .modal-header h3 { font-size: 1.3rem; font-weight: 800; color: #0B1023; }
+ .modal-plan { font-size: 0.85rem; color: #888; margin-top: 0.2rem; }
+ .modal-close { background: none; border: none; font-size: 1.2rem; color: #aaa; cursor: pointer; padding: 0.25rem; }
+ .modal-close:hover { color: #0B1023; }
+
+ .modal-row { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }
+ .modal-field { display: flex; flex-direction: column; gap: 0.35rem; margin-bottom: 1rem; }
+ .modal-field label { font-size: 0.85rem; font-weight: 600; color: #444; }
+ .modal-field input, .modal-field textarea {
+ border: 1.5px solid #ddd; border-radius: 8px;
+ padding: 0.65rem 0.9rem; font-size: 0.95rem;
+ font-family: inherit; resize: vertical;
+ transition: border-color 0.15s;
+ }
+ .modal-field input:focus, .modal-field textarea:focus { outline: none; border-color: #F97316; }
+ .modal-error { color: #dc2626; font-size: 0.85rem; background: #fef2f2; padding: 0.6rem; border-radius: 8px; margin-bottom: 1rem; }
+
+ .modal-success { text-align: center; padding: 1rem 0; }
+ .modal-success-icon { width: 3.5rem; height: 3.5rem; background: #dcfce7; color: #16a34a; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-size: 1.5rem; margin: 0 auto 1rem; }
+ .modal-success h3 { font-size: 1.3rem; font-weight: 800; color: #0B1023; margin-bottom: 0.5rem; }
+ .modal-success p { color: #666; margin-bottom: 1.5rem; }
+
+ @media (max-width: 480px) { .modal-row { grid-template-columns: 1fr; } }
+
footer .container {
display: flex;
justify-content: space-between;
diff --git a/app/src/routes/admin/+layout.svelte b/app/src/routes/admin/+layout.svelte
index ebf934a..b070c44 100644
--- a/app/src/routes/admin/+layout.svelte
+++ b/app/src/routes/admin/+layout.svelte
@@ -3,6 +3,11 @@
import { goto } from '$app/navigation';
import { pb } from '$lib/pb';
import logo from '$lib/assets/checkflo-logo.png';
+ import iconDashboard from '$lib/assets/icons/dashboard.svg';
+ import iconLogs from '$lib/assets/icons/logs.svg';
+ import iconProtokoll from '$lib/assets/icons/protokoll.svg';
+ import iconStations from '$lib/assets/icons/stations.svg';
+ import iconLogout from '$lib/assets/icons/logout.svg';
let { children } = $props();
@@ -14,9 +19,10 @@
}
const navItems = [
- { href: '/admin/dashboard', label: 'Dashboard', icon: '◈' },
- { href: '/admin/logs', label: 'Protokoll', icon: '≡' },
- { href: '/admin/stations', label: 'Stationen', icon: '⊞' }
+ { href: '/admin/dashboard', label: 'Dashboard', icon: iconDashboard },
+ { href: '/admin/logs', label: 'Einträge', icon: iconLogs },
+ { href: '/admin/protokoll', label: 'Protokoll', icon: iconProtokoll },
+ { href: '/admin/stations', label: 'Stationen', icon: iconStations }
];
@@ -35,12 +41,15 @@
class="nav-item"
class:active={$page.url.pathname.startsWith(item.href)}
>
-
{item.icon}
+
{item.label}
{/each}
-
Abmelden
+
+
+ Abmelden
+
@@ -84,20 +93,39 @@
}
.nav-item:hover { background: rgba(255,255,255,0.06); color: #fff; }
.nav-item.active { background: rgba(249,115,22,0.15); color: #F97316; }
- .nav-icon { font-size: 1.1rem; }
+
+ .nav-icon {
+ width: 18px;
+ height: 18px;
+ flex-shrink: 0;
+ filter: brightness(0) saturate(100%) invert(55%) sepia(14%) saturate(400%) hue-rotate(182deg);
+ }
+ .nav-item.active .nav-icon {
+ filter: brightness(0) saturate(100%) invert(56%) sepia(85%) saturate(600%) hue-rotate(359deg);
+ }
+ .nav-item:hover .nav-icon {
+ filter: brightness(0) invert(1);
+ }
.logout {
+ display: flex;
+ align-items: center;
+ gap: 0.75rem;
background: none;
border: 1px solid #2a3347;
color: #8892a4;
- padding: 0.6rem;
+ padding: 0.6rem 0.75rem;
border-radius: 8px;
cursor: pointer;
font-size: 0.85rem;
transition: all 0.15s;
margin-top: 1rem;
+ width: 100%;
}
.logout:hover { border-color: #F97316; color: #F97316; }
+ .logout:hover .nav-icon {
+ filter: brightness(0) saturate(100%) invert(56%) sepia(85%) saturate(600%) hue-rotate(359deg);
+ }
main {
flex: 1;
@@ -105,6 +133,12 @@
padding: 2rem;
}
+ @media print {
+ aside { display: none !important; }
+ :global(.shell) { display: block !important; }
+ main { padding: 0 !important; overflow: visible !important; }
+ }
+
@media (max-width: 640px) {
.shell { flex-direction: column; }
aside {
@@ -115,8 +149,9 @@
}
.logo-link { margin-bottom: 0; margin-right: auto; }
nav { flex-direction: row; flex: 0; gap: 0.5rem; }
- .nav-item span:last-child { display: none; }
- .logout { margin-top: 0; padding: 0.5rem 0.75rem; }
+ .nav-item span { display: none; }
+ .logout { margin-top: 0; padding: 0.5rem 0.75rem; width: auto; }
+ .logout :global(span) { display: none; }
main { padding: 1rem; }
}
diff --git a/app/src/routes/admin/dashboard/+page.svelte b/app/src/routes/admin/dashboard/+page.svelte
index bd32c37..d6906f2 100644
--- a/app/src/routes/admin/dashboard/+page.svelte
+++ b/app/src/routes/admin/dashboard/+page.svelte
@@ -24,7 +24,9 @@
const dateStr = today.toISOString().slice(0, 19).replace('T', ' ');
const result = await pb.collection('check_logs').getList(1, 50, {
- sort: '-id'
+ filter: `tenant = '${tenantId}' && created >= '${dateStr}'`,
+ expand: 'station',
+ sort: '-created'
});
logs = result.items;
@@ -40,7 +42,10 @@
});
function formatTime(iso: string) {
- return new Date(iso).toLocaleTimeString('de-DE', { hour: '2-digit', minute: '2-digit' });
+ if (!iso) return '—';
+ const d = new Date(iso);
+ if (isNaN(d.getTime())) return '—';
+ return d.toLocaleTimeString('de-DE', { hour: '2-digit', minute: '2-digit' });
}
diff --git a/app/src/routes/admin/logs/+page.svelte b/app/src/routes/admin/logs/+page.svelte
index c4c1cbb..74d8224 100644
--- a/app/src/routes/admin/logs/+page.svelte
+++ b/app/src/routes/admin/logs/+page.svelte
@@ -25,7 +25,7 @@
const result = await pb.collection('check_logs').getList(p, PER_PAGE, {
filter: `tenant = '${tenantId}'`,
expand: 'station',
- sort: '-id'
+ sort: '-created'
});
logs = result.items;
page = p;
@@ -38,7 +38,10 @@
}
function formatDate(iso: string) {
- return new Date(iso).toLocaleString('de-DE', {
+ if (!iso) return '—';
+ const d = new Date(iso);
+ if (isNaN(d.getTime())) return '—';
+ return d.toLocaleString('de-DE', {
day: '2-digit', month: '2-digit', year: '2-digit',
hour: '2-digit', minute: '2-digit'
});
diff --git a/app/src/routes/admin/protokoll/+page.svelte b/app/src/routes/admin/protokoll/+page.svelte
new file mode 100644
index 0000000..6a4cf0b
--- /dev/null
+++ b/app/src/routes/admin/protokoll/+page.svelte
@@ -0,0 +1,259 @@
+
+
+Protokoll {months[selectedMonth-1]} {selectedYear} — checkflo
+
+
+
+
+
Monatsprotokoll
+
+
+
+ {#each months as m, i}
+ {m}
+ {/each}
+
+
+ {#each [2025, 2026, 2027] as y}
+ {y}
+ {/each}
+
+
window.print()}>
+ Drucken / PDF
+
+
+
+
+
+
+
+
+ {#if loading}
+
Lädt…
+ {:else if logs.length === 0}
+
Keine Einträge im gewählten Zeitraum.
+
Keine Einträge im {months[selectedMonth-1]} {selectedYear}.
+ {:else}
+ {#each Object.entries(byStation()) as [stationName, stationLogs]}
+
+
+
+
+
+ Datum / Zeit
+ Temp. (°C)
+ Status
+ Notiz
+ Geprüft von
+
+
+
+ {#each stationLogs as log}
+
+ {formatDate(log.created)}
+ {log.temperature ? log.temperature + ' °C' : '—'}
+
+
+ {STATUS_LABEL[log.status] ?? log.status}
+
+
+ {log.notes || '—'}
+ {log.checked_by}
+
+ {/each}
+
+
+
+ {/each}
+
+
+ Gesamt: {logs.length} Einträge ·
+ {logs.filter(l=>l.status==='ok').length} OK ·
+ {logs.filter(l=>l.status==='abweichung').length} Abweichungen ·
+ {logs.filter(l=>l.status==='kritisch').length} Kritisch
+
+ {/if}
+
+
+
+
+
diff --git a/app/src/routes/admin/stations/+page.svelte b/app/src/routes/admin/stations/+page.svelte
index c46836d..bd21112 100644
--- a/app/src/routes/admin/stations/+page.svelte
+++ b/app/src/routes/admin/stations/+page.svelte
@@ -1,6 +1,8 @@
Stationen — checkflo
-
-
Stationen
-
Jeden QR-Code ausdrucken und an der Station befestigen. Der Link öffnet die Checkliste direkt im Browser.
+
+
{#if loading}
Lädt…
@@ -53,29 +107,19 @@
{#each stations as s}
-
-
+
+
{s.name}
{TYPE_LABEL[s.type] ?? s.type}
{#if s.target_temp_min || s.target_temp_max}
· {s.target_temp_min}° bis {s.target_temp_max}°C
{/if}
+
{qrUrl(s.qr_id)}
-
-
-
-
{qrUrl(s.qr_id)}
-
copyUrl(s.qr_id)}>
- {copied === s.qr_id ? '✓ Kopiert' : 'Kopieren'}
-
-
- QR öffnen
-
+ {#if qrCodes[s.qr_id]}
+
+ {/if}
{/each}
@@ -83,65 +127,105 @@
{/if}
+
+{#if !loading}
+
+ {#each stations as s}
+
+
{tenantName}
+
{s.name}
+
{TYPE_LABEL[s.type] ?? s.type}
+ {#if s.target_temp_min || s.target_temp_max}
+ · Zielbereich {s.target_temp_min}° bis {s.target_temp_max}°C
+ {/if}
+
+ {#if qrCodes[s.qr_id]}
+
+ {/if}
+
{qrUrl(s.qr_id)}
+
+
+ {/each}
+
+{/if}
+
diff --git a/app/src/routes/api/contact/+server.ts b/app/src/routes/api/contact/+server.ts
new file mode 100644
index 0000000..8114736
--- /dev/null
+++ b/app/src/routes/api/contact/+server.ts
@@ -0,0 +1,44 @@
+import { json } from '@sveltejs/kit';
+
+export async function POST({ request }) {
+ const body = await request.json();
+ const { token, name, email, company, phone, message, plan } = body;
+
+ if (!name?.trim() || !email?.trim()) {
+ return json({ error: 'Name und E-Mail sind Pflichtfelder.' }, { status: 400 });
+ }
+
+ const secret = process.env.TURNSTILE_SECRET ?? '';
+
+ // Turnstile serverseitig verifizieren
+ const verification = await fetch('https://challenges.cloudflare.com/turnstile/v0/siteverify', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ secret, response: token })
+ });
+ const result = await verification.json();
+
+ if (!result.success) {
+ return json({ error: 'Captcha-Verifizierung fehlgeschlagen.' }, { status: 400 });
+ }
+
+ // Anfrage in PocketBase speichern
+ const pb = await fetch('https://api.checkflo.de/api/collections/inquiries/records', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({
+ name: name.trim(),
+ email: email.trim(),
+ company: company?.trim() ?? '',
+ phone: phone?.trim() ?? '',
+ message: message?.trim() ?? '',
+ plan: plan ?? ''
+ })
+ });
+
+ if (!pb.ok) {
+ return json({ error: 'Fehler beim Speichern.' }, { status: 500 });
+ }
+
+ return json({ success: true });
+}
diff --git a/app/src/routes/s/[id]/+page.svelte b/app/src/routes/s/[id]/+page.svelte
index 6b496f3..1a991f6 100644
--- a/app/src/routes/s/[id]/+page.svelte
+++ b/app/src/routes/s/[id]/+page.svelte
@@ -85,7 +85,7 @@