Compare commits

...

4 Commits

38 changed files with 4557 additions and 2962 deletions

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" fill="#eaeaea" viewBox="0 0 256 256"><path d="M224,128a8,8,0,0,1-8,8H59.31l58.35,58.34a8,8,0,0,1-11.32,11.32l-72-72a8,8,0,0,1,0-11.32l72-72a8,8,0,0,1,11.32,11.32L59.31,120H216A8,8,0,0,1,224,128Z"></path></svg>

After

Width:  |  Height:  |  Size: 273 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" fill="#eaeaea" viewBox="0 0 256 256"><path d="M216,48H40A16,16,0,0,0,24,64V224a15.84,15.84,0,0,0,9.25,14.5A16.05,16.05,0,0,0,40,240a15.89,15.89,0,0,0,10.25-3.78l.09-.07L83,208H216a16,16,0,0,0,16-16V64A16,16,0,0,0,216,48ZM40,224h0ZM216,192H80a8,8,0,0,0-5.23,1.95L40,224V64H216Z"></path></svg>

After

Width:  |  Height:  |  Size: 355 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" fill="#eaeaea" viewBox="0 0 256 256"><path d="M205.66,194.34a8,8,0,0,1-11.32,11.32L128,139.31,61.66,205.66a8,8,0,0,1-11.32-11.32L116.69,128,50.34,61.66A8,8,0,0,1,61.66,50.34L128,116.69l66.34-66.35a8,8,0,0,1,11.32,11.32L139.31,128Z"></path></svg>

After

Width:  |  Height:  |  Size: 309 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 48 KiB

View File

@ -1,4 +1,11 @@
import { spawn } from "node:child_process"; import { spawn } from "node:child_process";
import path from "node:path";
const viteBin = path.join(
"node_modules",
".bin",
process.platform === "win32" ? "vite.cmd" : "vite"
);
const run = (name, command, args, env = {}) => { const run = (name, command, args, env = {}) => {
const child = spawn(command, args, { const child = spawn(command, args, {
@ -22,7 +29,7 @@ const run = (name, command, args, env = {}) => {
}; };
const api = run("api", "node", ["server/index.js"], { API_PORT: "4174" }); const api = run("api", "node", ["server/index.js"], { API_PORT: "4174" });
const vite = run("vite", "node_modules/.bin/vite", []); const vite = run("vite", viteBin, []);
const stop = () => { const stop = () => {
api.kill("SIGTERM"); api.kill("SIGTERM");

View File

@ -13,14 +13,67 @@ body,
#root { #root {
background: var(--theme-bg); background: var(--theme-bg);
color: var(--theme-text); color: var(--theme-text);
min-width: 0;
} }
body { body {
background: var(--theme-bg); background: var(--theme-bg);
} }
main {
min-width: 0;
}
.shell {
width: var(--container-wide);
margin: 0 auto;
padding: 0 0 clamp(2.2rem, 6vw, 5rem);
background: transparent;
border: 0;
box-shadow: none;
}
.page,
.detail-page,
.discovery-page,
.about-page,
.support-page,
.small-page,
.impressum-page,
.datenschutz-page {
isolation: isolate;
}
.detail-page .navbar--light,
.discovery-page .navbar--light,
.about-page .navbar--light,
.support-page .navbar--light,
.small-page .navbar--light,
.impressum-page .navbar--light,
.datenschutz-page .navbar--light {
position: fixed;
top: clamp(0.75rem, 2.1vw, 1.4rem);
right: 0;
left: 0;
z-index: 998;
margin-bottom: 0;
padding-top: 0;
pointer-events: none;
}
.detail-page .navbar--light .nav-pill,
.discovery-page .navbar--light .nav-pill,
.about-page .navbar--light .nav-pill,
.support-page .navbar--light .nav-pill,
.small-page .navbar--light .nav-pill,
.impressum-page .navbar--light .nav-pill,
.datenschutz-page .navbar--light .nav-pill {
pointer-events: auto;
}
.navbar--light .nav-pill, .navbar--light .nav-pill,
.navbar--light .nav-link, .navbar--light .nav-link,
.shell,
[class*="-page"], [class*="-page"],
[class*="-shell"], [class*="-shell"],
[class*="-card"], [class*="-card"],
@ -29,7 +82,11 @@ body {
button, button,
input, input,
textarea { textarea {
transition: background-color 0.24s ease, border-color 0.24s ease, color 0.24s ease; transition:
background-color var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
} }
body.theme-dark .navbar--light .nav-pill { body.theme-dark .navbar--light .nav-pill {

View File

@ -13,6 +13,7 @@ import SupportChatbot from "./components/SupportChatbot";
import ScrollToTop from "./components/ScrollToTop"; import ScrollToTop from "./components/ScrollToTop";
import ShopDrawer from "./components/ShopDrawer"; import ShopDrawer from "./components/ShopDrawer";
import CartToast from "./components/CartToast"; import CartToast from "./components/CartToast";
import { ProductTransitionProvider } from "./components/ProductTransition";
import useScrollTextReveal from "./hooks/useScrollTextReveal"; import useScrollTextReveal from "./hooks/useScrollTextReveal";
import { ThemeProvider } from "./theme/ThemeContext"; import { ThemeProvider } from "./theme/ThemeContext";
import "./style/textReveal.css"; import "./style/textReveal.css";
@ -30,7 +31,7 @@ function App() {
const shouldFlushFooter = const shouldFlushFooter =
location.pathname === "/" || location.pathname.startsWith("/duft/"); location.pathname === "/" || location.pathname.startsWith("/duft/");
useScrollTextReveal(routeContentRef, [location.pathname]); useScrollTextReveal(routeContentRef, location.pathname);
useEffect(() => { useEffect(() => {
if (typeof document === "undefined") return; if (typeof document === "undefined") return;
@ -52,10 +53,10 @@ function App() {
return ( return (
<ThemeProvider value={{ theme, isLight, toggleTheme }}> <ThemeProvider value={{ theme, isLight, toggleTheme }}>
<> <ProductTransitionProvider>
<ScrollToTop /> <ScrollToTop />
<div ref={routeContentRef}> <div ref={routeContentRef} data-route-content>
<Routes> <Routes>
<Route path="/" element={<LandingPage />} /> <Route path="/" element={<LandingPage />} />
<Route path="/duft/:perfumeSlug" element={<ProductDetailPage />} /> <Route path="/duft/:perfumeSlug" element={<ProductDetailPage />} />
@ -72,7 +73,7 @@ function App() {
<CartToast /> <CartToast />
<Footer flushTop={shouldFlushFooter} /> <Footer flushTop={shouldFlushFooter} />
<SupportChatbot /> <SupportChatbot />
</> </ProductTransitionProvider>
</ThemeProvider> </ThemeProvider>
); );
} }

View File

@ -1,6 +1,10 @@
.site-footer { .site-footer {
margin-top: 40px; position: relative;
background: #1f1f1f; margin-top: 0;
overflow: hidden;
background:
radial-gradient(circle at 82% 10%, rgba(var(--theme-accent-rgb) / 0.15), transparent 22rem),
#171717;
color: #f5f5f5; color: #f5f5f5;
border-top: 1px solid rgba(255, 255, 255, 0.08); border-top: 1px solid rgba(255, 255, 255, 0.08);
} }
@ -9,76 +13,102 @@
margin-top: 0; margin-top: 0;
} }
.site-footer::before {
content: "ATMOS";
position: absolute;
right: var(--page-x);
bottom: -0.16em;
color: rgba(255, 255, 255, 0.035);
font-size: clamp(5.5rem, 18vw, 20rem);
line-height: 0.8;
letter-spacing: 0;
pointer-events: none;
}
.site-footer__inner { .site-footer__inner {
max-width: 1600px; position: relative;
z-index: 1;
width: var(--container-wide);
margin: 0 auto; margin: 0 auto;
padding: 28px 20px 32px; padding: clamp(2.4rem, 7vw, 6.5rem) 0 clamp(2.2rem, 5vw, 4.8rem);
display: grid; display: grid;
grid-template-columns: 1.4fr 1fr 1fr; grid-template-columns: minmax(0, 1.4fr) minmax(12rem, 0.65fr) minmax(12rem, 0.75fr);
gap: 28px; gap: var(--gap-lg);
align-items: start;
} }
.site-footer__brand { .site-footer__brand {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 12px; gap: var(--gap-sm);
} }
.site-footer__logo { .site-footer__logo {
text-decoration: none; width: fit-content;
color: #fff; color: #fff;
font-size: 14px; font-size: clamp(1rem, 1.5vw, 1.2rem);
letter-spacing: 0.22em; letter-spacing: 0.22em;
text-decoration: none;
} }
.site-footer__text { .site-footer__text {
max-width: 32rem;
margin: 0; margin: 0;
max-width: 320px; color: rgba(255, 255, 255, 0.7);
font-size: 14px; font-size: var(--text-base);
line-height: 1.6; line-height: 1.65;
color: rgba(255, 255, 255, 0.72);
} }
.site-footer__nav-group { .site-footer__nav-group {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 12px; gap: var(--gap-sm);
padding-top: 0.2rem;
} }
.site-footer__heading { .site-footer__heading {
font-size: 10px; color: rgba(255, 255, 255, 0.52);
font-size: var(--text-xs);
letter-spacing: 0.22em; letter-spacing: 0.22em;
text-transform: uppercase; text-transform: uppercase;
color: rgba(255, 255, 255, 0.5);
} }
.site-footer__nav { .site-footer__nav {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 10px; gap: 0.72rem;
} }
.site-footer__nav a { .site-footer__nav a {
text-decoration: none; width: fit-content;
color: #f5f5f5; color: #f5f5f5;
font-size: 14px; font-size: var(--text-sm);
transition: opacity 0.2s ease, transform 0.2s ease; line-height: 1.2;
text-decoration: none;
transition:
color var(--duration-med) var(--ease-out),
opacity var(--duration-med) var(--ease-out),
transform var(--duration-med) var(--ease-out);
} }
.site-footer__nav a:hover { .site-footer__nav a:hover,
opacity: 0.7; .site-footer__nav a:focus-visible {
transform: translateX(2px); color: var(--theme-accent);
transform: translateX(0.25rem);
} }
@media (max-width: 900px) { @media (max-width: 900px) {
.site-footer__inner { .site-footer__inner {
grid-template-columns: 1fr 1fr; grid-template-columns: 1fr 1fr;
} }
.site-footer__brand {
grid-column: 1 / -1;
}
} }
@media (max-width: 640px) { @media (max-width: 640px) {
.site-footer__inner { .site-footer__inner {
grid-template-columns: 1fr; grid-template-columns: 1fr;
padding: 24px 16px 28px;
} }
} }

File diff suppressed because it is too large Load Diff

View File

@ -2,272 +2,53 @@ import { useEffect, useMemo, useState } from "react";
import { Link, useNavigate, useParams } from "react-router"; import { Link, useNavigate, useParams } from "react-router";
import perfumes from "../data/perfumes"; import perfumes from "../data/perfumes";
import SharedNavbar from "./SharedNavbar"; import SharedNavbar from "./SharedNavbar";
import { useProductTransition } from "../transitions/ProductTransitionContext";
import { formatChf } from "../shop/money"; import { formatChf } from "../shop/money";
import { useShop } from "../shop/useShop"; import { useShop } from "../shop/useShop";
import "./ProductDetailPage.css"; import "./ProductDetailPage.css";
const STORY_PANEL_IMAGE = "/placeholder-character-panel.jpg";
const priceToCents = (price) => { const priceToCents = (price) => {
const match = String(price).match(/(\d+)/); const match = String(price).match(/(\d+)/);
return match ? Number(match[1]) * 100 : 0; return match ? Number(match[1]) * 100 : 0;
}; };
function ProductDetailContent({ perfumeSlug }) { function ProductPurchasePanel({
const navigate = useNavigate(); perfume,
const { addToCart, subscribeToProduct, user } = useShop(); selectedSize,
setSelectedSize,
const perfume = useMemo( selectedPriceCents,
() => perfumes.find((item) => item.slug === perfumeSlug) || perfumes[0], discountPreviewCents,
[perfumeSlug] addToCart,
); subscribeToProduct,
}) {
const [selectedImage, setSelectedImage] = useState(
perfume.gallery?.[0] || perfume.image
);
const [selectedSize, setSelectedSize] = useState("sample");
const [showReviewDetails, setShowReviewDetails] = useState(false);
const [commentPage, setCommentPage] = useState(0);
const [isStructureOpen, setIsStructureOpen] = useState(false);
const [isMoodOpen, setIsMoodOpen] = useState(false);
const selectedProductId = `${perfume.slug}-${selectedSize === "sample" ? "sample" : "full"}`; const selectedProductId = `${perfume.slug}-${selectedSize === "sample" ? "sample" : "full"}`;
const selectedProductLabel = selectedSize === "sample" ? "Sample" : "Full Size"; const selectedProductLabel = selectedSize === "sample" ? "Sample" : "Full Size";
const selectedPriceCents = priceToCents(perfume.prices[selectedSize]);
const sampleCredit = user?.sampleCredits?.find(
(credit) => credit.slug === perfume.slug && credit.status === "available"
);
const discountPreviewCents =
selectedSize === "full"
? Math.min(
selectedPriceCents,
(user?.discoveryStatus === "Discount available" ? 4800 : 0) +
(sampleCredit?.amount_cents || 0)
)
: 0;
const sizeOptions = [ const sizeOptions = [
{ {
key: "sample", key: "sample",
title: "Sample 2ml", title: "Sample 2ml",
price: perfume.prices.sample, price: perfume.prices.sample,
note: "Zum Testen · ca. 20 Anwendungen", note: "Zum Testen, ca. 20 Anwendungen",
}, },
{ {
key: "full", key: "full",
title: "Full Size 50ml", title: "Full Size 50ml",
price: perfume.prices.full, price: perfume.prices.full,
note: "Nachkauf · 500+ Anwendungen", note: "Nachkauf, 500+ Anwendungen",
}, },
]; ];
const reviewSummary = perfume.reviews || {
score: 0,
total: 0,
metrics: [],
};
const reviewComments = perfume.commentSpotlight || [];
const commentPages = [];
for (let i = 0; i < reviewComments.length; i += 2) {
commentPages.push(reviewComments.slice(i, i + 2));
}
const safeCommentPages =
commentPages.length > 0
? commentPages
: [
[
{
id: "fallback-1",
name: "Atelier",
title: "Noch keine Stimmen",
text: "Für diesen Duft sind aktuell noch keine Kommentare hinterlegt.",
},
],
];
useEffect(() => {
const interval = window.setInterval(() => {
setCommentPage((prev) => (prev + 1) % safeCommentPages.length);
}, 5000);
return () => window.clearInterval(interval);
}, [safeCommentPages.length]);
return ( return (
<div className="detail-page"> <aside className="product-purchase-panel" data-product-transition-reveal>
<SharedNavbar variant="light" /> <div className="purchase-price-row">
<span>Preis</span>
<main className="detail-shell"> <strong>{perfume.prices[selectedSize]}</strong>
<div className="detail-topbar">
<button className="back-link" type="button" onClick={() => navigate("/")}>
<span className="back-link-arrow"></span>
<span>Zurück zur Startseite</span>
</button>
<div className="detail-topbar-meta">
<span className="detail-topbar-label">DUFTDETAIL</span>
<span className="detail-topbar-name">{perfume.name}</span>
</div>
</div> </div>
<section className="detail-layout"> <div className="purchase-size-group">
<div className="detail-gallery"> <span className="label-title">Größe wählen</span>
<div className="detail-main-image">
<img src={selectedImage} alt={perfume.name} />
</div>
<div className="detail-thumbs">
{[perfume.image, ...(perfume.gallery || [])]
.slice(0, 3)
.map((img, index) => (
<button
key={`${img}-${index}`}
type="button"
className={`thumb-btn ${selectedImage === img ? "active" : ""}`}
onClick={() => setSelectedImage(img)}
>
<img src={img} alt={`${perfume.name} Ansicht ${index + 1}`} />
</button>
))}
</div>
<div className="detail-meta-grid detail-meta-grid--top">
<div>
<span>TRAGEHINWEIS</span>
<p>{perfume.dosage}</p>
</div>
<div>
<span>HALTBARKEIT</span>
<p>{perfume.longevity}</p>
</div>
<div>
<span>ANLASS</span>
<p>{perfume.occasion}</p>
</div>
</div>
{/* --- Accordion Group Start --- */}
<div className="accordion-group">
{/* Dropdown: Duftstruktur */}
<div className={`accordion-item ${isStructureOpen ? "is-open" : ""}`}>
<button
type="button"
className="accordion-toggle"
onClick={() => setIsStructureOpen(!isStructureOpen)}
>
<span>DUFTSTRUKTUR</span>
<span className="accordion-icon">{isStructureOpen ? "" : "+"}</span>
</button>
{isStructureOpen && (
<div className="accordion-content">
<div className="detail-structure-layout">
<div className="detail-structure">
<div className="structure-block">
<span className="structure-phase">PHASE 1: TOP NOTES (01 H)</span>
<div className="structure-tags-grid">
{perfume.phases.top.map((note) => (
<span key={note} className="structure-tag">{note}</span>
))}
</div>
</div>
<div className="structure-block">
<span className="structure-phase">PHASE 2: HEART NOTES (14 H)</span>
<div className="structure-tags-grid">
{perfume.phases.heart.map((note) => (
<span key={note} className="structure-tag">{note}</span>
))}
</div>
</div>
<div className="structure-block">
<span className="structure-phase">PHASE 3: BASE NOTES (4 H+)</span>
<div className="structure-tags-grid">
{perfume.phases.base.map((note) => (
<span key={note} className="structure-tag">{note}</span>
))}
</div>
</div>
</div>
<aside className="structure-info-box">
<span className="structure-info-label">ZUR EINORDNUNG</span>
<p>
Die Duftstruktur zeigt, wie sich der Duft über die Zeit entfaltet:
der erste Eindruck im Auftakt, die eigentliche Signatur im Herzen
und die Spur, die lange auf Haut und Kleidung bleibt.
</p>
<div className="structure-info-legend">
<div>
<span className="structure-info-dot structure-info-dot--light" />
<span>Auftakt</span>
</div>
<div>
<span className="structure-info-dot structure-info-dot--mid" />
<span>Herz</span>
</div>
<div>
<span className="structure-info-dot structure-info-dot--strong" />
<span>Basis</span>
</div>
</div>
</aside>
</div>
</div>
)}
</div>
{/* Dropdown: Moodsetting */}
<div className={`accordion-item ${isMoodOpen ? "is-open" : ""}`}>
<button
type="button"
className="accordion-toggle"
onClick={() => setIsMoodOpen(!isMoodOpen)}
>
<span>MOODSETTING</span>
<span className="accordion-icon">{isMoodOpen ? "" : "+"}</span>
</button>
{isMoodOpen && (
<div className="accordion-content">
<div className="mood-box-content">
<p>{perfume.mood}</p>
</div>
</div>
)}
</div>
</div>
{/* --- Accordion Group End --- */}
</div>
<div className="detail-info">
<div className="detail-heading" data-reveal-group>
<div className="detail-heading-copy">
<span className="detail-kicker" data-reveal="fade">
Edition 04
</span>
<h1 data-reveal="fade">{perfume.name}</h1>
<p data-reveal="fade">{perfume.shortText}</p>
</div>
</div>
<div className="detail-section-block">
<span className="label-title">MATERIAL-KOMPOSITION</span>
<div className="material-tags">
{perfume.materialTags.map((tag) => (
<span key={tag}>{tag}</span>
))}
</div>
</div>
<div className="detail-section-block">
<span className="label-title">GRÖSSE WÄHLEN</span>
<div className="size-grid"> <div className="size-grid">
{sizeOptions.map((option) => ( {sizeOptions.map((option) => (
<button <button
@ -275,6 +56,7 @@ function ProductDetailContent({ perfumeSlug }) {
type="button" type="button"
className={`size-card ${selectedSize === option.key ? "active" : ""}`} className={`size-card ${selectedSize === option.key ? "active" : ""}`}
onClick={() => setSelectedSize(option.key)} onClick={() => setSelectedSize(option.key)}
aria-pressed={selectedSize === option.key}
> >
<span className="size-title">{option.title}</span> <span className="size-title">{option.title}</span>
<strong>{option.price}</strong> <strong>{option.price}</strong>
@ -284,26 +66,7 @@ function ProductDetailContent({ perfumeSlug }) {
</div> </div>
</div> </div>
<div className="discovery-note" data-reveal-group data-reveal-start="top 88%"> <div className="purchase-actions">
<div className="discovery-note-text" data-reveal="fade">
<strong>Discovery Set wird einmalig angerechnet</strong>
<p>
Nur das erste Discovery Set erzeugt CHF 48 Guthaben. Es wird
einmal bei einem späteren Full-Size-Kauf automatisch abgezogen.
</p>
{discountPreviewCents > 0 && (
<p className="discount-preview">
Erwarteter Preis mit Rabatt:{" "}
<strong>{formatChf(selectedPriceCents - discountPreviewCents)}</strong>
</p>
)}
</div>
<Link to="/discovery-set" className="discovery-note-btn" data-reveal="fade">
Zum Set
</Link>
</div>
<button <button
className="buy-button" className="buy-button"
type="button" type="button"
@ -315,7 +78,7 @@ function ProductDetailContent({ perfumeSlug }) {
).catch(() => {}) ).catch(() => {})
} }
> >
KAUFEN Kaufen
</button> </button>
<button <button
@ -323,63 +86,252 @@ function ProductDetailContent({ perfumeSlug }) {
type="button" type="button"
onClick={() => subscribeToProduct(selectedProductId, "restock").catch(() => {})} onClick={() => subscribeToProduct(selectedProductId, "restock").catch(() => {})}
> >
RESTOCK UPDATE ABONNIEREN Restock Update abonnieren
</button> </button>
<div className="detail-description-section">
<span className="label-title">BESCHREIBUNG</span>
<div className="detail-columns">
<div className="detail-copy-block">
<span className="detail-copy-label">PARFÜMERIE / STUDIO</span>
<p>{perfume.description}</p>
</div> </div>
<div className="detail-copy-block"> <div className="purchase-discovery-note">
<span className="detail-copy-label">HERKUNFT</span> <div>
<p>{perfume.origin}</p> <strong>Discovery Set wird angerechnet</strong>
<p>
Sample- und Set-Guthaben werden beim Full-Size-Kauf automatisch
abgezogen.
</p>
{discountPreviewCents > 0 && (
<p className="discount-preview">
Erwarteter Preis mit Rabatt:{" "}
<strong>{formatChf(selectedPriceCents - discountPreviewCents)}</strong>
</p>
)}
</div>
<Link to="/discovery-set">Zum Set</Link>
</div>
</aside>
);
}
function ProductHero({
perfume,
selectedImage,
setSelectedImage,
selectedSize,
setSelectedSize,
selectedPriceCents,
discountPreviewCents,
addToCart,
subscribeToProduct,
}) {
const galleryImages = [...new Set([perfume.image, ...(perfume.gallery || [])])].slice(0, 3);
return (
<section className="product-hero">
<div className="product-media-column">
<div className="product-hero-copy" data-product-transition-reveal>
<span>Edition {perfume.id}</span>
<h1>{perfume.name}</h1>
</div> </div>
<div className="detail-copy-block"> <div className="product-hero-visual">
<span className="detail-copy-label">KONZENTRATION</span> <img
src={selectedImage}
alt={perfume.name}
className="product-hero-image"
data-product-transition-target={perfume.slug}
decoding="async"
/>
</div>
<div className="product-thumbs" data-product-transition-reveal>
{galleryImages.map((img, index) => (
<button
key={`${img}-${index}`}
type="button"
className={`thumb-btn ${selectedImage === img ? "active" : ""}`}
onClick={() => setSelectedImage(img)}
aria-label={`${perfume.name} Ansicht ${index + 1}`}
>
<img src={img} alt="" loading="lazy" decoding="async" />
</button>
))}
</div>
<div className="hero-fact-grid" data-product-transition-reveal>
<div>
<span>Tragehinweis</span>
<p>{perfume.dosage}</p>
</div>
<div>
<span>Konzentration</span>
<p>{perfume.concentration}</p> <p>{perfume.concentration}</p>
</div> </div>
<div>
<div className="detail-copy-block"> <span>Studio</span>
<span className="detail-copy-label">EDITION</span> <p>{perfume.description}</p>
<p>{perfume.edition}</p>
</div> </div>
</div> </div>
</div> </div>
<div className="delivery-box"> <ProductPurchasePanel
<div className="delivery-box-header"> perfume={perfume}
<span className="label-title">LIEFERUNG</span> selectedSize={selectedSize}
setSelectedSize={setSelectedSize}
selectedPriceCents={selectedPriceCents}
discountPreviewCents={discountPreviewCents}
addToCart={addToCart}
subscribeToProduct={subscribeToProduct}
/>
</section>
);
}
function ProductStorySection({ perfume }) {
return (
<section className="product-story-grid" data-reveal-group>
<div className="story-copy">
<span className="eyebrow" data-reveal="fade">Beschreibung / Charakter</span>
<h2 data-reveal="lines">{perfume.text}</h2>
<p data-reveal="fade">{perfume.mood}</p>
<div className="character-facts" data-reveal="fade">
<div>
<span>Haltbarkeit</span>
<p>{perfume.longevity}</p>
</div>
<div>
<span>Anlass</span>
<p>{perfume.occasion}</p>
</div>
<div>
<span>Lieferung</span>
<p>Versand in 1-2 Werktagen. Zustellung in der Regel in 5-6 Tagen.</p>
</div>
</div>
</div>
<div className="story-visual-panel" data-reveal="fade">
<img
className="story-visual-image"
src={STORY_PANEL_IMAGE}
alt=""
loading="lazy"
decoding="async"
/>
<div className="story-visual-content">
<span>Material-Komposition</span>
<div className="material-tags">
{perfume.materialTags.map((tag) => (
<span key={tag}>{tag}</span>
))}
</div>
</div>
</div>
</section>
);
}
function ProductStructureSection({ perfume }) {
const phaseGroups = [
{ label: "Top Notes / 0-1 h", notes: perfume.phases.top },
{ label: "Heart Notes / 1-4 h", notes: perfume.phases.heart },
{ label: "Base Notes / 4 h+", notes: perfume.phases.base },
];
return (
<section className="product-structure-section" data-reveal-group>
<div className="section-intro">
<span className="eyebrow" data-reveal="fade">Duftstruktur</span>
<h2 data-reveal="lines">Die Spur entwickelt sich in Schichten.</h2>
</div>
<div className="structure-timeline">
{phaseGroups.map((phase, index) => (
<article className="structure-phase-card" key={phase.label} data-reveal="fade">
<span className="structure-index">0{index + 1}</span>
<h3>{phase.label}</h3>
<div>
{phase.notes.map((note) => (
<span key={note}>{note}</span>
))}
</div>
</article>
))}
</div>
</section>
);
}
function ProductMetaSection({ perfume }) {
const metaItems = [
["Parfümerie / Studio", perfume.description],
["Herkunft", perfume.origin],
["Konzentration", perfume.concentration],
["Edition", perfume.edition],
];
return (
<section className="product-meta-section" data-reveal-group>
<div className="meta-grid">
{metaItems.map(([label, value]) => (
<div className="meta-card" key={label} data-reveal="fade">
<span>{label}</span>
<p>{value}</p>
</div>
))}
</div>
<div className="delivery-panel" data-reveal="fade">
<div className="delivery-panel-header">
<span className="eyebrow">Lieferung</span>
<span className="delivery-badge">CH</span> <span className="delivery-badge">CH</span>
</div> </div>
<div className="delivery-grid"> <div className="delivery-grid">
<div className="delivery-item"> <div>
<span className="delivery-item-label">VERSAND</span> <span>Versand</span>
<p>Innerhalb von 12 Werktagen</p> <p>Innerhalb von 1-2 Werktagen</p>
</div> </div>
<div>
<div className="delivery-item"> <span>Zustellung</span>
<span className="delivery-item-label">ZUSTELLUNG</span> <p>In der Regel in 5-6 Tagen bei dir</p>
<p>In der Regel in 56 Tagen bei dir</p>
</div> </div>
<div>
<div className="delivery-item delivery-item--full"> <span>Hinweis</span>
<span className="delivery-item-label">HINWEIS</span>
<p>Sorgfältig verpackt und geschützt versendet.</p> <p>Sorgfältig verpackt und geschützt versendet.</p>
</div> </div>
</div> </div>
</div> </div>
</section>
);
}
<div className="comment-spotlight"> function ProductReviews({
<div className="comment-spotlight-header"> reviewSummary,
<span className="label-title">STIMMEN ZUM DUFT</span> safeCommentPages,
commentPage,
setCommentPage,
showReviewDetails,
setShowReviewDetails,
}) {
return (
<section className="product-reviews-section" data-reveal-group>
<div className="reviews-heading" data-reveal="fade">
<div>
<span className="eyebrow">Stimmen zum Duft</span>
<h2>Resonanz</h2>
<p>
Verdichtete Wahrnehmung aus bisherigen Stimmen zu Charakter,
Haltbarkeit, Sillage und Originalität.
</p>
</div>
<div className="review-score-block">
<strong>{reviewSummary.score.toFixed(1)}</strong>
<span></span>
<small>{reviewSummary.total} Stimmen</small>
</div>
</div>
<div className="comment-spotlight" data-reveal="fade">
<div className="comment-dots"> <div className="comment-dots">
{safeCommentPages.map((_, index) => ( {safeCommentPages.map((_, index) => (
<button <button
@ -391,39 +343,19 @@ function ProductDetailContent({ perfumeSlug }) {
/> />
))} ))}
</div> </div>
</div>
<div className="comment-spotlight-grid"> <div className="comment-spotlight-grid">
{safeCommentPages[commentPage].map((comment) => ( {safeCommentPages[commentPage].map((comment) => (
<article className="comment-card" key={comment.id}> <article className="comment-card" key={comment.id}>
<span className="comment-card-title">{comment.title}</span> <span>{comment.title}</span>
<p>{comment.text}</p> <p>{comment.text}</p>
<span className="comment-card-author">{comment.name}</span> <small>{comment.name}</small>
</article> </article>
))} ))}
</div> </div>
</div> </div>
<div className="review-section"> <div className={`review-panel ${showReviewDetails ? "is-open" : ""}`} data-reveal="fade">
<div className="review-section-header">
<div className="review-section-copy">
<span className="label-title">RESONANZ</span>
<p className="review-section-text">
Verdichtete Wahrnehmung aus bisherigen Stimmen zu Charakter,
Haltbarkeit, Sillage und Originalität.
</p>
</div>
<div className="review-section-main">
<span className="review-score">{reviewSummary.score.toFixed(1)}</span>
<div className="review-summary-copy">
<span className="review-stars"></span>
<span className="review-count">{reviewSummary.total} Stimmen</span>
</div>
</div>
</div>
<div className={`review-panel ${showReviewDetails ? "is-open" : ""}`}>
<button <button
type="button" type="button"
className="review-toggle" className="review-toggle"
@ -437,64 +369,217 @@ function ProductDetailContent({ perfumeSlug }) {
</button> </button>
{showReviewDetails && ( {showReviewDetails && (
<div className="review-popover">
<div className="review-details"> <div className="review-details">
{reviewSummary.metrics.map((metric) => ( {reviewSummary.metrics.map((metric) => (
<div className="review-detail-row" key={metric.label}> <div className="review-detail-row" key={metric.label}>
<span className="review-detail-label">{metric.label}</span> <span>{metric.label}</span>
<div className="review-detail-bar"> <div className="review-detail-bar">
<div <div
className="review-detail-fill" className="review-detail-fill"
style={{ width: `${(metric.value / 5) * 100}%` }} style={{ width: `${(metric.value / 5) * 100}%` }}
/> />
</div> </div>
<span className="review-detail-value"> <strong>{metric.value.toFixed(1)}</strong>
{metric.value.toFixed(1)}
</span>
</div> </div>
))} ))}
<button <button type="button" className="review-write-button" disabled>
type="button"
className="review-write-button"
disabled
title="Später mit Login verfügbar"
>
Bewertung schreiben Bewertung schreiben
</button> </button>
</div> </div>
</div>
)} )}
</div> </div>
</div>
</div>
</section> </section>
);
}
function ProductTestingCTA({ perfume, addToCart }) {
return (
<section className="detail-bottom-cta" data-reveal-group> <section className="detail-bottom-cta" data-reveal-group>
<h2 data-reveal="fade">Lieber erst testen?</h2> <div>
<span className="eyebrow" data-reveal="fade">Lieber erst testen?</span>
<h2 data-reveal="lines">Sample oder Discovery Set.</h2>
<p data-reveal="fade"> <p data-reveal="fade">
Bestelle ein 2ml Sample für CHF 12 oder das komplette Discovery Set Bestelle ein 2ml Sample für CHF 12 oder das komplette Discovery Set mit
mit allen 6 Düften für CHF 48. Beide werden beim späteren Full-Size-Kauf allen 6 Düften für CHF 48. Beide werden beim späteren Full-Size-Kauf
vollständig angerechnet. vollständig angerechnet.
</p> </p>
</div>
<div className="detail-bottom-actions" data-reveal="fade"> <div className="detail-bottom-actions" data-reveal="fade">
<button <button
type="button" type="button"
onClick={() => onClick={() =>
addToCart(`${perfume.slug}-sample`, 1, `${perfume.name} Sample added.`).catch(() => {}) addToCart(`${perfume.slug}-sample`, 1, `${perfume.name} Sample added.`).catch(
() => {}
)
} }
> >
SAMPLE BESTELLEN {perfume.prices.sample} Sample bestellen - {perfume.prices.sample}
</button> </button>
<button <button
type="button" type="button"
onClick={() => addToCart("discovery-set", 1, "Discovery Set added.").catch(() => {})} onClick={() => addToCart("discovery-set", 1, "Discovery Set added.").catch(() => {})}
> >
DISCOVERY SET CHF 48 Discovery Set - CHF 48
</button> </button>
</div> </div>
</section> </section>
);
}
function ProductRecommendations({ currentSlug, startProductTransition }) {
const recommendations = perfumes
.filter((item) => item.slug !== currentSlug)
.slice(0, 3);
return (
<section className="recommendation-section" data-reveal-group>
<div className="recommendation-heading">
<span className="eyebrow" data-reveal="fade">Empfehlungen</span>
<h2 data-reveal="lines">Weitere Atmosphären</h2>
</div>
<div className="recommendation-grid">
{recommendations.map((item) => (
<Link
to={`/duft/${item.slug}`}
className="recommendation-card"
key={item.slug}
onClick={(event) => startProductTransition(event, item)}
data-reveal="fade"
>
<span>{item.id}</span>
<img
src={item.image}
alt={item.name}
loading="lazy"
decoding="async"
data-product-transition-source
/>
<div>
<h3>{item.name}</h3>
<p>{item.text}</p>
</div>
</Link>
))}
</div>
</section>
);
}
function ProductDetailContent({ perfumeSlug }) {
const navigate = useNavigate();
const { addToCart, subscribeToProduct, user } = useShop();
const { activeSlug, phase, startProductTransition } = useProductTransition();
const perfume = useMemo(
() => perfumes.find((item) => item.slug === perfumeSlug) || perfumes[0],
[perfumeSlug]
);
const [selectedImage, setSelectedImage] = useState(
perfume.gallery?.[0] || perfume.image
);
const [selectedSize, setSelectedSize] = useState("sample");
const [showReviewDetails, setShowReviewDetails] = useState(false);
const [commentPage, setCommentPage] = useState(0);
const selectedPriceCents = priceToCents(perfume.prices[selectedSize]);
const sampleCredit = user?.sampleCredits?.find(
(credit) => credit.slug === perfume.slug && credit.status === "available"
);
const discountPreviewCents =
selectedSize === "full"
? Math.min(
selectedPriceCents,
(user?.discoveryStatus === "Discount available" ? 4800 : 0) +
(sampleCredit?.amount_cents || 0)
)
: 0;
const reviewSummary = perfume.reviews || {
score: 0,
total: 0,
metrics: [],
};
const safeCommentPages = useMemo(() => {
const reviewComments = perfume.commentSpotlight || [];
const pages = [];
for (let i = 0; i < reviewComments.length; i += 2) {
pages.push(reviewComments.slice(i, i + 2));
}
return pages.length > 0
? pages
: [
[
{
id: "fallback-1",
name: "Atelier",
title: "Noch keine Stimmen",
text: "Für diesen Duft sind aktuell noch keine Kommentare hinterlegt.",
},
],
];
}, [perfume.commentSpotlight]);
const isTransitionArriving = activeSlug === perfume.slug && phase === "entering";
useEffect(() => {
const interval = window.setInterval(() => {
setCommentPage((prev) => (prev + 1) % safeCommentPages.length);
}, 5000);
return () => window.clearInterval(interval);
}, [safeCommentPages.length]);
return (
<div className={`detail-page ${isTransitionArriving ? "is-transition-arriving" : ""}`}>
<SharedNavbar variant="hero" />
<main className="shell">
<div className="detail-topbar" data-product-transition-reveal>
<button className="back-link" type="button" onClick={() => navigate("/")}>
<span className="back-link-arrow" aria-hidden="true" />
<span>Zurück zur Startseite</span>
</button>
<div className="detail-topbar-meta">
<span>Duftdetail</span>
<strong>{perfume.name}</strong>
</div>
</div>
<ProductHero
perfume={perfume}
selectedImage={selectedImage}
setSelectedImage={setSelectedImage}
selectedSize={selectedSize}
setSelectedSize={setSelectedSize}
selectedPriceCents={selectedPriceCents}
discountPreviewCents={discountPreviewCents}
addToCart={addToCart}
subscribeToProduct={subscribeToProduct}
/>
<ProductStorySection perfume={perfume} />
<ProductStructureSection perfume={perfume} />
<ProductMetaSection perfume={perfume} />
<ProductReviews
reviewSummary={reviewSummary}
safeCommentPages={safeCommentPages}
commentPage={commentPage}
setCommentPage={setCommentPage}
showReviewDetails={showReviewDetails}
setShowReviewDetails={setShowReviewDetails}
/>
<ProductTestingCTA perfume={perfume} addToCart={addToCart} />
<ProductRecommendations
currentSlug={perfume.slug}
startProductTransition={startProductTransition}
/>
</main> </main>
</div> </div>
); );

View File

@ -0,0 +1,37 @@
.product-transition {
position: fixed;
inset: 0;
z-index: 2500;
pointer-events: none;
opacity: 0;
visibility: hidden;
overflow: hidden;
}
.product-transition__wash {
position: absolute;
inset: 0;
background: var(--theme-bg);
}
.product-transition__image {
position: absolute;
top: 0;
left: 0;
display: block;
max-width: none;
object-fit: contain;
transform-origin: 0 0;
will-change: transform, opacity;
filter: drop-shadow(0 34px 82px rgba(0, 0, 0, 0.42));
}
body.product-transition-active {
cursor: progress;
}
@media (prefers-reduced-motion: reduce) {
.product-transition {
display: none;
}
}

View File

@ -0,0 +1,336 @@
import {
useCallback,
useLayoutEffect,
useMemo,
useRef,
useState,
} from "react";
import { useNavigate } from "react-router";
import { gsap } from "gsap";
import { ProductTransitionContext } from "../transitions/ProductTransitionContext";
import "./ProductTransition.css";
const supportsProductTransition = () => {
if (typeof window === "undefined") return false;
return !window.matchMedia("(prefers-reduced-motion: reduce)").matches;
};
const rectToObject = (rect) => ({
top: rect.top,
left: rect.left,
width: rect.width,
height: rect.height,
});
const isPlainNavigationClick = (event) =>
event.button === 0 &&
!event.metaKey &&
!event.altKey &&
!event.ctrlKey &&
!event.shiftKey;
const getStageRect = (sourceRect) => {
const viewportWidth = window.innerWidth;
const viewportHeight = window.innerHeight;
const sourceRatio = sourceRect.height / sourceRect.width || 1;
const stageWidth = Math.min(
Math.max(sourceRect.width * 1.16, 260),
viewportWidth * 0.56,
620
);
const stageHeight = Math.min(stageWidth * sourceRatio, viewportHeight * 0.72);
const normalizedWidth = stageHeight / sourceRatio;
return {
left: (viewportWidth - normalizedWidth) / 2,
top: (viewportHeight - stageHeight) / 2,
width: normalizedWidth,
height: stageHeight,
};
};
const getTransformForRect = (rect, sourceRect) => {
const scale = Math.min(
rect.width / sourceRect.width,
rect.height / sourceRect.height
);
const width = sourceRect.width * scale;
const height = sourceRect.height * scale;
return {
x: rect.left + (rect.width - width) / 2,
y: rect.top + (rect.height - height) / 2,
scaleX: scale,
scaleY: scale,
};
};
export function ProductTransitionProvider({ children }) {
const navigate = useNavigate();
const overlayRef = useRef(null);
const imageRef = useRef(null);
const washRef = useRef(null);
const timelineRef = useRef(null);
const [transition, setTransition] = useState(null);
const clearTransition = useCallback(() => {
timelineRef.current?.kill();
timelineRef.current = null;
document.body.classList.remove("product-transition-active");
setTransition(null);
}, []);
const startProductTransition = useCallback(
(event, perfume) => {
if (!perfume?.slug || !isPlainNavigationClick(event)) {
return false;
}
if (!supportsProductTransition()) {
return false;
}
const card = event.currentTarget;
const image = card.querySelector("[data-product-transition-source]");
if (!image) {
return false;
}
const sourceRect = image.getBoundingClientRect();
if (sourceRect.width <= 0 || sourceRect.height <= 0) {
return false;
}
event.preventDefault();
timelineRef.current?.kill();
document.body.classList.add("product-transition-active");
setTransition({
id: `${perfume.slug}-${Date.now()}`,
slug: perfume.slug,
to: `/duft/${perfume.slug}`,
image: image.currentSrc || image.src || perfume.image,
alt: perfume.name,
sourceRect: rectToObject(sourceRect),
phase: "leaving",
});
return true;
},
[]
);
useLayoutEffect(() => {
if (transition?.phase !== "leaving") {
return undefined;
}
const overlay = overlayRef.current;
const image = imageRef.current;
const wash = washRef.current;
if (!overlay || !image || !wash) {
return undefined;
}
const routeContent = document.querySelector("[data-route-content]");
const sourceRect = transition.sourceRect;
const stageRect = getStageRect(sourceRect);
let completed = false;
gsap.set(overlay, { autoAlpha: 1, pointerEvents: "auto" });
gsap.set(wash, { autoAlpha: 0 });
gsap.set(image, {
x: sourceRect.left,
y: sourceRect.top,
width: sourceRect.width,
height: sourceRect.height,
scaleX: 1,
scaleY: 1,
autoAlpha: 1,
transformOrigin: "0 0",
force3D: true,
});
timelineRef.current = gsap.timeline({
defaults: { ease: "power4.inOut" },
onComplete: () => {
completed = true;
navigate(transition.to, {
state: {
productTransition: true,
transitionId: transition.id,
},
});
setTransition((current) =>
current?.id === transition.id
? { ...current, phase: "entering" }
: current
);
},
});
timelineRef.current
.to(wash, { autoAlpha: 1, duration: 0.42, ease: "power2.out" }, 0)
.to(
routeContent,
{
autoAlpha: 0,
filter: "none",
scale: 1,
duration: 0.62,
ease: "power3.out",
},
0
)
.to(
image,
{
...getTransformForRect(stageRect, sourceRect),
duration: 0.78,
},
0.03
);
return () => {
if (!completed) {
timelineRef.current?.kill();
}
};
}, [navigate, transition]);
useLayoutEffect(() => {
if (transition?.phase !== "entering") {
return undefined;
}
const overlay = overlayRef.current;
const image = imageRef.current;
const wash = washRef.current;
if (!overlay || !image || !wash) {
return undefined;
}
let frame = 0;
let cancelled = false;
let completed = false;
const sourceRect = transition.sourceRect;
const runEnterAnimation = () => {
if (cancelled) return;
window.scrollTo({ top: 0, left: 0, behavior: "instant" });
const target = document.querySelector(
`[data-product-transition-target="${transition.slug}"]`
);
if (!target && frame < 16) {
frame += 1;
window.requestAnimationFrame(runEnterAnimation);
return;
}
const routeContent = document.querySelector("[data-route-content]");
gsap.set(routeContent, {
autoAlpha: 1,
filter: "none",
scale: 1,
clearProps: "opacity,visibility,filter,transform",
});
if (!target) {
gsap.to(overlay, {
autoAlpha: 0,
duration: 0.3,
ease: "power2.out",
onComplete: clearTransition,
});
return;
}
const targetRect = rectToObject(target.getBoundingClientRect());
const revealItems = gsap.utils.toArray("[data-product-transition-reveal]");
gsap.set(target, { autoAlpha: 0 });
gsap.set(revealItems, { y: 28, autoAlpha: 0, force3D: true });
timelineRef.current?.kill();
timelineRef.current = gsap.timeline({
defaults: { ease: "power4.out" },
onComplete: () => {
completed = true;
clearTransition();
},
});
timelineRef.current
.to(image, {
...getTransformForRect(targetRect, sourceRect),
duration: 0.72,
})
.set(target, { autoAlpha: 1 }, ">-0.08")
.to(image, { autoAlpha: 0, duration: 0.16, ease: "power2.out" }, "<")
.to(wash, { autoAlpha: 0, duration: 0.5, ease: "power2.out" }, "<")
.to(
revealItems,
{
y: 0,
autoAlpha: 1,
duration: 0.82,
stagger: 0.07,
ease: "power4.out",
clearProps: "transform",
},
">-0.05"
);
};
window.requestAnimationFrame(runEnterAnimation);
return () => {
cancelled = true;
if (!completed) {
timelineRef.current?.kill();
}
};
}, [clearTransition, transition]);
const value = useMemo(
() => ({
activeSlug: transition?.slug || null,
phase: transition?.phase || "idle",
startProductTransition,
}),
[startProductTransition, transition]
);
return (
<ProductTransitionContext.Provider value={value}>
{children}
{transition && (
<div
className={`product-transition product-transition--${transition.phase}`}
ref={overlayRef}
aria-hidden="true"
>
<div className="product-transition__wash" ref={washRef} />
<img
className="product-transition__image"
src={transition.image}
alt={transition.alt}
ref={imageRef}
decoding="async"
/>
</div>
)}
</ProductTransitionContext.Provider>
);
}

View File

@ -1,9 +1,9 @@
import { Link } from "react-router"; import { Link } from "react-router";
import { useShop } from "../shop/useShop"; import { useShop } from "../shop/useShop";
import { useTheme } from "../theme/ThemeContext"; import { useTheme } from "../theme/useTheme";
import "../style/navbar.css"; import "../style/navbar.css";
function SharedNavbar({ variant = "light", active = "" }) { function SharedNavbar({ variant = "hero", active = "" }) {
const { cart, openCart, openProfile, user } = useShop(); const { cart, openCart, openProfile, user } = useShop();
const { isLight, toggleTheme } = useTheme(); const { isLight, toggleTheme } = useTheme();
const cartLabel = const cartLabel =

View File

@ -495,3 +495,89 @@
} }
/* --- Design System Refinement Start --- */
.shop-drawer {
width: min(560px, 100%);
min-width: 0;
padding: clamp(1rem, 3vw, 1.4rem);
box-shadow: var(--theme-shadow);
}
.drawer-top button,
.cart-toast-close,
.cart-controls button,
.subscription-row button {
min-width: 44px;
min-height: 44px;
}
.drawer-section,
.cart-item,
.order-card,
.read-block,
.status-box,
.requirement-row,
.drawer-error,
.payment-card,
.pref-toggle,
.subscription-row,
.totals-box .discount-explainer,
.cart-toast,
.shop-field input {
border-radius: var(--radius-lg);
}
.drawer-primary,
.drawer-secondary,
.cart-remove,
.cart-controls,
.cart-controls button,
.cart-toast button,
.subscription-row button {
border-radius: 999px;
}
.drawer-primary,
.cart-toast > button:last-child {
min-height: 48px;
}
.drawer-primary,
.drawer-secondary,
.cart-remove,
.payment-card,
.pref-toggle,
.subscription-row button {
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
}
.drawer-primary:hover:not(:disabled),
.drawer-secondary:hover,
.cart-remove:hover,
.payment-card:hover,
.pref-toggle:hover,
.subscription-row button:hover {
transform: translateY(-1px);
border-color: rgba(var(--theme-accent-rgb) / 0.55);
}
@media (max-width: 640px) {
.shop-drawer {
width: 100%;
}
.cart-toast {
right: var(--page-x);
bottom: var(--page-x);
width: calc(100vw - (var(--page-x) * 2));
}
}
/* --- Design System Refinement End --- */

View File

@ -274,3 +274,118 @@
} }
} }
/* --- Design System Refinement Start --- */
.chatbot-trigger {
right: max(0.9rem, env(safe-area-inset-right));
bottom: max(0.9rem, env(safe-area-inset-bottom));
width: 52px;
height: 52px;
min-height: 52px;
min-width: 52px;
padding: 0;
display: inline-grid;
place-items: center;
font-size: 0;
box-shadow: var(--theme-shadow-soft);
}
.chatbot-trigger.is-open {
padding: 0;
}
.chatbot-trigger-icon,
.chatbot-close-icon {
display: block;
width: 1.45rem;
height: 1.45rem;
background: currentColor;
-webkit-mask-position: center;
mask-position: center;
-webkit-mask-repeat: no-repeat;
mask-repeat: no-repeat;
-webkit-mask-size: contain;
mask-size: contain;
}
.chatbot-trigger-icon {
-webkit-mask-image: url("/icon-chat.svg");
mask-image: url("/icon-chat.svg");
}
.chatbot-trigger.is-open .chatbot-trigger-icon,
.chatbot-close-icon {
-webkit-mask-image: url("/icon-x.svg");
mask-image: url("/icon-x.svg");
}
.chatbot-trigger:hover,
.chatbot-trigger:focus-visible {
transform: translateY(-2px);
box-shadow: var(--theme-shadow);
}
.chatbot-window {
right: var(--page-x);
bottom: calc(var(--page-x) + 4rem);
border-radius: var(--radius-lg);
box-shadow: var(--theme-shadow);
}
.chatbot-close,
.chatbot-send,
.chatbot-chip,
.chatbot-feedback-btn {
min-height: 44px;
}
.chatbot-close {
min-width: 44px;
}
.chatbot-message,
.chatbot-chip,
.chatbot-feedback-btn,
.chatbot-input,
.chatbot-send {
border-radius: var(--radius-lg);
}
.chatbot-send {
border-radius: 999px;
}
.chatbot-chip,
.chatbot-feedback-btn,
.chatbot-send {
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out);
}
.chatbot-chip:hover,
.chatbot-feedback-btn:hover,
.chatbot-send:hover {
transform: translateY(-1px);
border-color: rgba(var(--theme-accent-rgb) / 0.55);
}
@media (max-width: 700px) {
.chatbot-trigger {
right: max(0.75rem, env(safe-area-inset-right));
bottom: max(0.75rem, env(safe-area-inset-bottom));
}
.chatbot-window {
right: var(--page-x);
bottom: calc(var(--page-x) + 4.1rem);
width: calc(100vw - (var(--page-x) * 2));
height: min(72svh, 620px);
max-height: calc(100svh - 6rem);
}
}
/* --- Design System Refinement End --- */

View File

@ -216,7 +216,7 @@ function SupportChatbot() {
onClick={() => setIsOpen((prev) => !prev)} onClick={() => setIsOpen((prev) => !prev)}
aria-label={isOpen ? "Chat schliessen" : "Support Chat öffnen"} aria-label={isOpen ? "Chat schliessen" : "Support Chat öffnen"}
> >
{isOpen ? "×" : "Support"} <span className="chatbot-trigger-icon" aria-hidden="true" />
</button> </button>
{isOpen && ( {isOpen && (
@ -232,7 +232,7 @@ function SupportChatbot() {
onClick={() => setIsOpen(false)} onClick={() => setIsOpen(false)}
aria-label="Chat schliessen" aria-label="Chat schliessen"
> >
× <span className="chatbot-close-icon" aria-hidden="true" />
</button> </button>
</div> </div>

View File

@ -1,6 +1,5 @@
import { Link } from "react-router"; import { Link } from "react-router";
import IntroOverlay from "./IntroOverlay"; import IntroOverlay from "./IntroOverlay";
import SharedNavbar from "../SharedNavbar";
function HeroSection({ function HeroSection({
heroImageWrapRef, heroImageWrapRef,
@ -17,16 +16,15 @@ function HeroSection({
<div className="hero-media" ref={heroImageWrapRef}> <div className="hero-media" ref={heroImageWrapRef}>
<img <img
src="/atmos-hero-image.png" src="/atmos-hero-image.png"
alt="Atmos Hero" alt="Atmos perfume bottle in a material-led campaign scene"
className="hero-media__image" className="hero-media__image"
ref={heroImageRef} ref={heroImageRef}
loading="eager" loading="eager"
decoding="async"
fetchPriority="high" fetchPriority="high"
/> />
</div> </div>
<SharedNavbar variant="hero" active="atmos" />
<div className="hero-content"> <div className="hero-content">
<h1 className="hero-title"> <h1 className="hero-title">
<span className="hero-title-line"> <span className="hero-title-line">

View File

@ -57,7 +57,7 @@ const restoreRevealLines = (element) => {
delete element.dataset.revealOriginalHtml; delete element.dataset.revealOriginalHtml;
}; };
function useScrollTextReveal(scopeRef, deps = []) { function useScrollTextReveal(scopeRef, dependencyKey = "") {
useLayoutEffect(() => { useLayoutEffect(() => {
const scope = scopeRef.current; const scope = scopeRef.current;
@ -162,7 +162,7 @@ function useScrollTextReveal(scopeRef, deps = []) {
ctx.revert(); ctx.revert();
preparedElements.forEach((element) => restoreRevealLines(element)); preparedElements.forEach((element) => restoreRevealLines(element));
}; };
}, [scopeRef, ...deps]); }, [scopeRef, dependencyKey]);
} }
export default useScrollTextReveal; export default useScrollTextReveal;

View File

@ -3,6 +3,7 @@
--theme-black: #262626; --theme-black: #262626;
--theme-white: #eaeaea; --theme-white: #eaeaea;
--theme-accent: #ff6a00; --theme-accent: #ff6a00;
--theme-accent-rgb: 255 106 0;
--theme-bg: #262626; --theme-bg: #262626;
--theme-surface: #2f2f2f; --theme-surface: #2f2f2f;
--theme-surface-soft: #363636; --theme-surface-soft: #363636;
@ -10,6 +11,47 @@
--theme-text: #eaeaea; --theme-text: #eaeaea;
--theme-text-muted: #c8c8c8; --theme-text-muted: #c8c8c8;
--theme-border: #4a4a4a; --theme-border: #4a4a4a;
--theme-border-strong: rgba(234, 234, 234, 0.26);
--theme-shadow: 0 24px 70px rgba(0, 0, 0, 0.28);
--theme-shadow-soft: 0 16px 42px rgba(0, 0, 0, 0.18);
--page-x: clamp(1rem, 4vw, 5rem);
--section-y-xs: clamp(2rem, 5vw, 4.5rem);
--section-y-sm: clamp(3rem, 7vw, 7rem);
--section-y: clamp(4rem, 10vw, 10rem);
--section-y-lg: clamp(5rem, 14vw, 14rem);
--container: min(calc(100% - (var(--page-x) * 2)), 1440px);
--container-narrow: min(calc(100% - (var(--page-x) * 2)), 920px);
--container-wide: min(calc(100% - (var(--page-x) * 2)), 1680px);
--text-measure: 68ch;
--gap-2xs: clamp(0.35rem, 0.7vw, 0.65rem);
--gap-xs: clamp(0.5rem, 1vw, 0.875rem);
--gap-sm: clamp(0.75rem, 1.5vw, 1.25rem);
--gap-md: clamp(1rem, 2vw, 2rem);
--gap-lg: clamp(1.5rem, 4vw, 4rem);
--gap-xl: clamp(2rem, 6vw, 6rem);
--radius-xs: 0;
--radius-sm: 0;
--radius-md: 0;
--radius-lg: 0;
--radius-xl: 0;
--text-xs: clamp(0.75rem, 0.72rem + 0.15vw, 0.875rem);
--text-sm: clamp(0.875rem, 0.83rem + 0.2vw, 1rem);
--text-base: clamp(1rem, 0.95rem + 0.25vw, 1.125rem);
--text-lg: clamp(1.125rem, 1.05rem + 0.35vw, 1.375rem);
--text-xl: clamp(1.35rem, 1.15rem + 0.9vw, 2rem);
--text-2xl: clamp(1.75rem, 1.25rem + 2vw, 3.5rem);
--text-display: clamp(3.05rem, 10.5vw, 10.8rem);
--ease-out: cubic-bezier(0.22, 1, 0.36, 1);
--ease-snap: cubic-bezier(0.16, 1, 0.3, 1);
--duration-fast: 160ms;
--duration-med: 260ms;
--duration-slow: 720ms;
color: var(--theme-text); color: var(--theme-text);
background: var(--theme-bg); background: var(--theme-bg);
@ -46,6 +88,9 @@ body.theme-light {
--theme-text: #262626; --theme-text: #262626;
--theme-text-muted: #5f5f5f; --theme-text-muted: #5f5f5f;
--theme-border: #d6d6d6; --theme-border: #d6d6d6;
--theme-border-strong: rgba(38, 38, 38, 0.22);
--theme-shadow: 0 24px 70px rgba(38, 38, 38, 0.13);
--theme-shadow-soft: 0 16px 42px rgba(38, 38, 38, 0.1);
} }
a { a {
@ -56,3 +101,65 @@ button {
font: inherit; font: inherit;
} }
html {
overflow-x: hidden;
overflow-x: clip;
}
body {
overflow-x: hidden;
overflow-x: clip;
}
img,
picture,
video,
canvas,
svg {
max-width: 100%;
}
img,
video {
height: auto;
}
button,
a,
input,
textarea,
select {
-webkit-tap-highlight-color: transparent;
}
button,
[role="button"],
a {
touch-action: manipulation;
}
:focus-visible {
outline: 2px solid var(--theme-accent);
outline-offset: 4px;
}
::selection {
background: rgba(var(--theme-accent-rgb) / 0.35);
color: var(--theme-white);
}
@media (prefers-reduced-motion: reduce) {
html {
scroll-behavior: auto;
}
*,
*::before,
*::after {
animation-duration: 0.01ms;
animation-iteration-count: 1;
scroll-behavior: auto;
transition-duration: 0.01ms;
}
}

View File

@ -1,208 +1,282 @@
.about-page { .about-page {
min-height: 100vh; min-height: 100vh;
padding: 0 0 var(--section-y-sm);
color: var(--theme-text); color: var(--theme-text);
padding: 26px 38px 38px;
background: background:
linear-gradient(to right, rgba(0, 0, 0, 0.45), rgba(0, 0, 0, 0.1)), radial-gradient(circle at 86% 8%, rgba(var(--theme-accent-rgb) / 0.13), transparent 28rem),
linear-gradient(to bottom, rgba(0, 0, 0, 0.2), rgba(0, 0, 0, 0.45)); linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
}
.about-shell {
background: var(--theme-surface);
border: 1px solid var(--theme-border);
padding: 38px;
} }
.about-kicker, .about-kicker,
.about-label, .about-label,
.about-panel-label, .about-panel-label,
.about-origin-box span, .about-origin-box span,
.about-panel-meta span { .about-panel-meta span,
.about-method-points span {
display: block; display: block;
font-size: 10px;
letter-spacing: 0.22em;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-xs);
letter-spacing: 0.22em;
text-transform: uppercase;
} }
.about-hero { .about-hero {
display: grid; display: grid;
grid-template-columns: minmax(0, 1.35fr) minmax(280px, 0.8fr); grid-template-columns: minmax(0, 1.45fr) minmax(18rem, 0.72fr);
gap: 28px; gap: var(--gap-lg);
align-items: end; align-items: end;
padding-bottom: 36px; padding: clamp(2rem, 5vw, 5rem) 0 var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); border-bottom: 1px solid var(--theme-border);
} }
.about-hero-copy {
min-width: 0;
}
.about-hero-copy h1 { .about-hero-copy h1 {
margin: 14px 0 18px; max-width: 11.4ch;
font-size: 68px; margin: clamp(0.85rem, 2vw, 1.2rem) 0 clamp(1rem, 2vw, 1.35rem);
line-height: 0.92;
font-weight: 300;
letter-spacing: -0.05em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(3rem, 7.4vw, 8.8rem);
line-height: 0.9;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.about-intro { .about-intro {
max-width: 720px; max-width: var(--text-measure);
margin: 0; margin: 0;
font-size: 18px;
line-height: 1.65;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-lg);
line-height: 1.65;
} }
.about-hero-panel { .about-hero-panel {
padding: 24px; padding: clamp(1.25rem, 3vw, 2rem);
background: linear-gradient( border: 1px solid rgba(var(--theme-accent-rgb) / 0.2);
180deg, background:
rgba(255, 106, 0, 0.08), linear-gradient(135deg, rgba(var(--theme-accent-rgb) / 0.1), transparent 62%),
rgba(255, 106, 0, 0.03) var(--theme-surface-soft);
);
border: 1px solid rgba(255, 106, 0, 0.18);
} }
.about-hero-panel p { .about-hero-panel p {
margin: 10px 0 0; margin: 0.75rem 0 0;
font-size: 16px;
line-height: 1.6;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-base);
line-height: 1.62;
} }
.about-panel-meta { .about-panel-meta {
display: grid; display: grid;
grid-template-columns: 1fr; gap: var(--gap-sm);
gap: 16px; margin-top: clamp(1.2rem, 2.6vw, 2rem);
margin-top: 24px; padding-top: var(--gap-sm);
padding-top: 20px; border-top: 1px solid rgba(var(--theme-accent-rgb) / 0.2);
border-top: 1px solid rgba(255, 106, 0, 0.14);
} }
.about-panel-meta p, .about-panel-meta p,
.about-origin-box p { .about-origin-box p,
margin: 8px 0 0; .about-method-points p {
font-size: 14px; margin: 0.45rem 0 0;
line-height: 1.55;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-sm);
line-height: 1.55;
} }
.about-section { .about-section {
padding-top: 38px; padding-top: var(--section-y-sm);
} }
.about-section--split { .about-section--split,
.about-origin-section,
.about-method-section {
display: grid; display: grid;
grid-template-columns: minmax(260px, 0.7fr) minmax(0, 1.3fr); grid-template-columns: minmax(16rem, 0.72fr) minmax(0, 1.28fr);
gap: 28px; gap: var(--gap-lg);
align-items: start; align-items: start;
} }
.about-section-heading h2, .about-section-heading h2,
.about-origin-copy h2, .about-origin-copy h2,
.about-bottom-copy h2 { .about-bottom-copy h2,
margin: 10px 0 0; .about-method-copy h2 {
font-size: 42px; margin: 0.75rem 0 0;
line-height: 0.98;
font-weight: 300;
letter-spacing: -0.04em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(2.15rem, 5.2vw, 6rem);
line-height: 0.94;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.about-section-copy { .about-section-copy,
.about-method-points {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 18px; gap: var(--gap-sm);
} }
.about-section-copy p, .about-section-copy p,
.about-origin-copy p, .about-origin-copy p,
.about-bottom-copy p { .about-bottom-copy p,
.about-method-copy p,
.about-credential-card p,
.about-card p,
.about-proof-item p,
.about-trust-note p {
margin: 0; margin: 0;
font-size: 16px;
line-height: 1.7;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.7;
} }
.about-quote-block { .about-section-copy p + p,
margin-top: 38px; .about-origin-copy p + p {
padding: 32px 36px; margin-top: var(--gap-sm);
background: #1f1f1f;
border-left: 3px solid #ff6a00;
} }
.about-quote-block p { .about-proof-strip,
margin: 0; .about-grid-section,
font-size: 28px; .about-credentials-grid {
line-height: 1.3; display: grid;
font-weight: 300; gap: var(--gap-sm);
letter-spacing: -0.03em; margin-top: var(--section-y-sm);
color: #fff; }
max-width: 900px;
.about-proof-strip {
grid-template-columns: repeat(4, minmax(0, 1fr));
} }
.about-grid-section { .about-grid-section {
display: grid; grid-template-columns: repeat(3, minmax(0, 1fr));
grid-template-columns: repeat(3, 1fr);
gap: 18px;
margin-top: 38px;
} }
.about-card { .about-credentials-grid {
padding: 24px; grid-template-columns: repeat(2, minmax(0, 1fr));
background: var(--theme-bg); }
.about-card,
.about-proof-item,
.about-credential-card,
.about-origin-box,
.about-method-section,
.about-trust-note {
border: 1px solid var(--theme-border); border: 1px solid var(--theme-border);
min-height: 260px; background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0.008)),
var(--theme-surface-soft);
} }
.about-card h3 { .about-card,
margin: 14px 0 12px; .about-proof-item,
font-size: 24px; .about-credential-card {
line-height: 1.05; min-height: 100%;
font-weight: 400; padding: clamp(1.1rem, 2.4vw, 1.8rem);
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
}
.about-card:hover,
.about-proof-item:hover,
.about-credential-card:hover {
transform: translateY(-4px);
border-color: rgba(var(--theme-accent-rgb) / 0.42);
box-shadow: var(--theme-shadow-soft);
}
.about-card h3,
.about-credential-card h3 {
margin: 0.9rem 0 0.75rem;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-xl);
line-height: 1.08;
font-weight: 400;
letter-spacing: 0;
} }
.about-card p { .about-proof-item {
min-height: 9rem;
}
.about-proof-item p {
margin-top: 0.7rem;
}
.about-quote-block {
margin-top: var(--section-y-sm);
padding: clamp(1.4rem, 4vw, 3rem);
overflow: hidden;
border-left: 3px solid var(--theme-accent);
background:
radial-gradient(circle at 100% 0%, rgba(var(--theme-accent-rgb) / 0.18), transparent 18rem),
#171717;
}
.about-quote-block p {
max-width: 58rem;
margin: 0; margin: 0;
font-size: 15px; color: #fff;
line-height: 1.65; font-size: clamp(1.7rem, 4vw, 4.2rem);
color: var(--theme-text-muted); line-height: 1.08;
font-weight: 300;
letter-spacing: 0;
} }
.about-process-section,
.about-origin-section { .about-origin-section {
display: grid; margin-top: var(--section-y-sm);
grid-template-columns: minmax(0, 1.2fr) minmax(280px, 0.8fr); padding-top: var(--section-y-sm);
gap: 28px; border-top: 1px solid var(--theme-border);
margin-top: 38px; }
padding-top: 38px;
.about-method-section {
margin-top: var(--section-y-sm);
padding: clamp(1.25rem, 3vw, 2.4rem);
}
.about-method-points > div,
.about-origin-box > div {
padding-top: 1rem;
border-top: 1px solid var(--theme-border); border-top: 1px solid var(--theme-border);
} }
.about-origin-box { .about-origin-box {
display: flex; display: grid;
flex-direction: column; gap: var(--gap-sm);
gap: 18px; padding: clamp(1.1rem, 2.4vw, 1.8rem);
background: var(--theme-bg);
border: 1px solid var(--theme-border);
padding: 24px;
} }
.about-origin-box > div { .about-trust-note {
padding-bottom: 16px; margin-top: var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); padding: clamp(1.1rem, 2.4vw, 1.8rem);
border-color: rgba(var(--theme-accent-rgb) / 0.24);
background:
linear-gradient(135deg, rgba(var(--theme-accent-rgb) / 0.11), transparent 60%),
var(--theme-surface-soft);
} }
.about-origin-box > div:last-child { .about-trust-note p {
border-bottom: none; max-width: 72rem;
padding-bottom: 0; margin-top: 0.75rem;
color: var(--theme-text);
} }
.about-bottom-cta { .about-bottom-cta {
margin-top: 38px;
padding: 38px;
background: #ff6a00;
display: grid; display: grid;
grid-template-columns: minmax(0, 1.2fr) auto; grid-template-columns: minmax(0, 1fr) auto;
gap: 24px; gap: var(--gap-lg);
align-items: end; align-items: end;
margin-top: var(--section-y-sm);
padding: clamp(1.5rem, 4vw, 3.5rem);
overflow: hidden;
background:
radial-gradient(circle at 92% 0%, rgba(255, 255, 255, 0.22), transparent 20rem),
var(--theme-accent);
} }
.about-bottom-copy .about-label, .about-bottom-copy .about-label,
@ -211,246 +285,90 @@
color: #fff; color: #fff;
} }
.about-bottom-copy .about-label {
opacity: 0.85;
}
.about-bottom-copy p { .about-bottom-copy p {
margin-top: 16px; max-width: 48rem;
max-width: 700px; margin-top: 1rem;
} }
.about-bottom-actions { .about-bottom-actions {
display: flex; display: flex;
gap: 12px;
flex-wrap: wrap; flex-wrap: wrap;
gap: var(--gap-xs);
justify-content: flex-end;
} }
.about-btn { .about-btn {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
text-decoration: none; min-height: 48px;
padding: 12px 18px; padding: 0 1.1rem;
font-size: 14px; border: 1px solid transparent;
border-radius: 999px; border-radius: 999px;
transition: transform 0.2s ease, opacity 0.2s ease, box-shadow 0.2s ease; color: inherit;
font-size: var(--text-sm);
text-decoration: none;
transition:
transform var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out);
} }
.about-btn:hover { .about-btn:hover,
transform: translateY(-1px); .about-btn:focus-visible {
transform: translateY(-2px);
box-shadow: var(--theme-shadow-soft);
} }
.about-btn--primary { .about-btn--primary {
background: var(--theme-paper); background: #fff;
color: #ff6a00; color: var(--theme-accent);
} }
.about-btn--secondary { .about-btn--secondary {
border-color: rgba(255, 255, 255, 0.22);
background: rgba(255, 255, 255, 0.14); background: rgba(255, 255, 255, 0.14);
color: #fff; color: #fff;
backdrop-filter: blur(8px); backdrop-filter: blur(8px);
} }
.about-proof-strip { @media (max-width: 1180px) {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 14px;
margin-top: 38px;
}
.about-proof-item {
padding: 18px;
border: 1px solid var(--theme-border);
background: var(--theme-surface-soft);
}
.about-proof-item p {
margin: 10px 0 0;
font-size: 14px;
line-height: 1.55;
color: var(--theme-text-muted);
}
.about-process-section {
margin-top: 38px;
padding-top: 38px;
border-top: 1px solid var(--theme-border);
}
.about-credentials-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 18px;
margin-top: 38px;
}
.about-credential-card {
padding: 24px;
background: linear-gradient(
180deg,
rgba(255, 106, 0, 0.06),
rgba(255, 106, 0, 0.02)
);
border: 1px solid rgba(255, 106, 0, 0.16);
min-height: 220px;
}
.about-credential-card h3 {
margin: 14px 0 12px;
font-size: 24px;
line-height: 1.08;
font-weight: 400;
color: var(--theme-text);
}
.about-credential-card p {
margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text-muted);
}
.about-method-section {
display: grid;
grid-template-columns: minmax(0, 1fr) minmax(320px, 0.95fr);
gap: 28px;
margin-top: 38px;
padding: 28px;
background: var(--theme-bg);
border: 1px solid var(--theme-border);
}
.about-method-copy h2 {
margin: 10px 0 16px;
font-size: 40px;
line-height: 0.98;
font-weight: 300;
letter-spacing: -0.04em;
color: var(--theme-text);
}
.about-method-copy p {
margin: 0;
font-size: 16px;
line-height: 1.7;
color: var(--theme-text-muted);
}
.about-method-points {
display: grid;
grid-template-columns: 1fr;
gap: 16px;
}
.about-method-points > div {
padding-bottom: 14px;
border-bottom: 1px solid var(--theme-border);
}
.about-method-points > div:last-child {
border-bottom: none;
padding-bottom: 0;
}
.about-method-points span {
display: block;
font-size: 10px;
letter-spacing: 0.22em;
color: var(--theme-text-muted);
}
.about-method-points p {
margin: 8px 0 0;
font-size: 14px;
line-height: 1.55;
color: var(--theme-text);
}
.about-trust-note {
margin-top: 38px;
padding: 22px;
border: 1px solid rgba(255, 106, 0, 0.18);
background: linear-gradient(
180deg,
rgba(255, 106, 0, 0.08),
rgba(255, 106, 0, 0.03)
);
}
.about-trust-note p {
margin: 10px 0 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text);
}
@media (max-width: 1100px) {
.about-hero, .about-hero,
.about-section--split, .about-section--split,
.about-origin-section, .about-origin-section,
.about-bottom-cta, .about-method-section,
.about-method-section { .about-bottom-cta {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.about-hero-copy h1 { .about-proof-strip,
font-size: 52px; .about-grid-section,
.about-credentials-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
}
@media (max-width: 760px) {
.about-hero {
padding-top: clamp(1.4rem, 5vw, 2rem);
}
.about-hero-copy h1 {
font-size: clamp(2.55rem, 13vw, 4.4rem);
} }
.about-grid-section,
.about-proof-strip, .about-proof-strip,
.about-grid-section,
.about-credentials-grid { .about-credentials-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.about-method-copy h2 { .about-bottom-actions {
font-size: 32px; display: grid;
} justify-content: stretch;
} }
@media (max-width: 700px) { .about-btn {
.about-page { width: 100%;
padding: 18px;
}
.about-shell {
padding: 24px 18px;
}
.about-hero-copy h1 {
font-size: 38px;
}
.about-section-heading h2,
.about-origin-copy h2,
.about-bottom-copy h2,
.about-method-copy h2 {
font-size: 30px;
}
.about-intro,
.about-section-copy p,
.about-origin-copy p,
.about-bottom-copy p,
.about-method-copy p,
.about-credential-card p,
.about-proof-item p,
.about-trust-note p {
font-size: 15px;
}
.about-quote-block {
padding: 24px 20px;
}
.about-quote-block p {
font-size: 22px;
}
.about-method-section,
.about-bottom-cta {
padding: 26px 20px;
} }
} }

View File

@ -5,9 +5,9 @@ import "./AboutPage.css";
function AboutPage() { function AboutPage() {
return ( return (
<div className="about-page"> <div className="about-page">
<SharedNavbar variant="light" /> <SharedNavbar variant="hero" />
<main className="about-shell"> <main className="shell">
<section className="about-hero" data-reveal-group> <section className="about-hero" data-reveal-group>
<div className="about-hero-copy"> <div className="about-hero-copy">
<span className="about-kicker" data-reveal="fade"> <span className="about-kicker" data-reveal="fade">

View File

@ -1,135 +1,108 @@
.datenschutz-page { .datenschutz-page {
min-height: 100vh; min-height: 100vh;
padding: 0 0 var(--section-y-sm);
color: var(--theme-text); color: var(--theme-text);
padding: 26px 38px 38px;
background: background:
linear-gradient(to right, rgba(0, 0, 0, 0.45), rgba(0, 0, 0, 0.1)), radial-gradient(circle at 86% 8%, rgba(var(--theme-accent-rgb) / 0.12), transparent 28rem),
linear-gradient(to bottom, rgba(0, 0, 0, 0.2), rgba(0, 0, 0, 0.45)); linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
}
.datenschutz-shell {
background: var(--theme-surface);
border: 1px solid var(--theme-border);
padding: 38px;
} }
.datenschutz-kicker, .datenschutz-kicker,
.datenschutz-label { .datenschutz-label {
display: block; display: block;
font-size: 10px;
letter-spacing: 0.22em;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-xs);
letter-spacing: 0.22em;
text-transform: uppercase;
} }
.datenschutz-hero { .datenschutz-hero {
padding-bottom: 32px; padding: clamp(2rem, 5vw, 5rem) 0 var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); border-bottom: 1px solid var(--theme-border);
} }
.datenschutz-hero h1 { .datenschutz-hero h1 {
margin: 14px 0 16px; margin: clamp(0.85rem, 2vw, 1.2rem) 0 clamp(1rem, 2vw, 1.35rem);
font-size: 64px;
line-height: 0.92;
font-weight: 300;
letter-spacing: -0.05em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(3rem, 8vw, 7.2rem);
line-height: 0.9;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
} }
.datenschutz-intro { .datenschutz-intro {
max-width: 820px; max-width: var(--text-measure);
margin: 0; margin: 0;
font-size: 17px;
line-height: 1.7;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-lg);
line-height: 1.7;
} }
.datenschutz-section { .datenschutz-section {
display: grid; display: grid;
grid-template-columns: minmax(260px, 0.7fr) minmax(0, 1.3fr); grid-template-columns: minmax(14rem, 0.72fr) minmax(0, 1.28fr);
gap: 28px; gap: var(--gap-lg);
align-items: start; align-items: start;
margin-top: 38px; margin-top: var(--section-y-sm);
padding-top: 38px; padding-top: var(--section-y-sm);
border-top: 1px solid var(--theme-border); border-top: 1px solid var(--theme-border);
} }
.datenschutz-section-heading h2 { .datenschutz-section-heading h2 {
margin: 10px 0 0; margin: 0.75rem 0 0;
font-size: 38px;
line-height: 0.98;
font-weight: 300;
letter-spacing: -0.04em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(2rem, 4.2vw, 4.5rem);
line-height: 0.96;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.datenschutz-section-copy p { .datenschutz-section-copy p {
margin: 0 0 16px; margin: 0 0 1rem;
font-size: 16px;
line-height: 1.75;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.75;
} }
.datenschutz-list { .datenschutz-list {
margin: 0;
padding-left: 18px;
display: grid; display: grid;
gap: 10px; gap: 0.75rem;
margin: 0;
padding-left: 1.1rem;
} }
.datenschutz-list li { .datenschutz-list li {
font-size: 16px;
line-height: 1.7;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.7;
} }
.datenschutz-note-box { .datenschutz-note-box {
padding: 22px; padding: clamp(1.1rem, 2.4vw, 1.8rem);
border: 1px solid rgba(255, 106, 0, 0.18); border: 1px solid rgba(var(--theme-accent-rgb) / 0.24);
background: linear-gradient( background:
180deg, linear-gradient(135deg, rgba(var(--theme-accent-rgb) / 0.11), transparent 60%),
rgba(255, 106, 0, 0.08), var(--theme-surface-soft);
rgba(255, 106, 0, 0.03)
);
} }
.datenschutz-note-box p { .datenschutz-note-box p {
margin: 0; margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-base);
line-height: 1.65;
} }
@media (max-width: 1100px) { @media (max-width: 900px) {
.datenschutz-section { .datenschutz-section {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.datenschutz-hero h1 {
font-size: 48px;
}
} }
@media (max-width: 700px) { @media (max-width: 700px) {
.datenschutz-page { .datenschutz-hero {
padding: 18px; padding-top: clamp(1.4rem, 5vw, 2rem);
}
.datenschutz-shell {
padding: 24px 18px;
}
.datenschutz-hero h1 {
font-size: 36px;
}
.datenschutz-section-heading h2 {
font-size: 28px;
}
.datenschutz-intro,
.datenschutz-section-copy p,
.datenschutz-list li {
font-size: 15px;
} }
} }

View File

@ -4,9 +4,9 @@ import "./DatenschutzPage.css";
function DatenschutzPage() { function DatenschutzPage() {
return ( return (
<div className="datenschutz-page"> <div className="datenschutz-page">
<SharedNavbar variant="light" /> <SharedNavbar variant="hero" />
<main className="datenschutz-shell"> <main className="shell">
<section className="datenschutz-hero" data-reveal-group> <section className="datenschutz-hero" data-reveal-group>
<span className="datenschutz-kicker" data-reveal="fade">RECHTLICHE ANGABEN</span> <span className="datenschutz-kicker" data-reveal="fade">RECHTLICHE ANGABEN</span>
<h1 data-reveal="lines">DATENSCHUTZ</h1> <h1 data-reveal="lines">DATENSCHUTZ</h1>

File diff suppressed because it is too large Load Diff

View File

@ -4,127 +4,203 @@ import SharedNavbar from "../components/SharedNavbar";
import { useShop } from "../shop/useShop"; import { useShop } from "../shop/useShop";
import "./DiscoverySetPage.css"; import "./DiscoverySetPage.css";
const moodImages = [ const DISCOVERY_SET_IMAGE = "/atmos-discovery-set-thumbnail.png";
"/DISCOVERYSET.png",
"/DISCOVERYSET.png", const discoveryPanelFacts = [
"/DISCOVERYSET.png", { label: "Umfang", value: "6 × 2ml" },
"/DISCOVERYSET.png", { label: "Gutschrift", value: "CHF 48 werden beim späteren Full-Size-Kauf berücksichtigt." },
"/DISCOVERYSET.png",
"/DISCOVERYSET.png",
]; ];
function DiscoverySetPage() { const discoveryBenefits = [
const navigate = useNavigate(); {
const { addToCart } = useShop(); title: "6 × 2ml Samples aller Signature-Düfte",
const buyDiscoverySet = () => text: "Kalter Beton, Schwarzes Benzin, Verbranntes Chrom, Blasse Seide, Weisse Asche und Nasser Marmor.",
addToCart("discovery-set", 1, "Discovery Set added.").catch(() => {}); },
{
title: "CHF 48 Gutschein automatisch im Set",
text: "Nur das erste Discovery Set erstellt den einmaligen Rabatt. Er wird bei einem späteren Full-Size-Kauf automatisch angerechnet.",
},
{
title: "Jedes Sample = ca. 20 Anwendungen",
text: "Genug, um jeden Duft mehrere Tage im Alltag und in unterschiedlichen Situationen zu testen.",
},
{
title: "Hochwertige Mini-Flacons",
text: "Sorgfältig zusammengestellt, reduziert gestaltet und als Teil des atmos Konzepts gedacht.",
},
];
const discoverySteps = [
{
number: "01",
title: "Bestellen",
text: "Discovery Set für CHF 48 bestellen. Nur dein erstes Set erzeugt automatisch einen einmaligen Rabatt.",
},
{
number: "02",
title: "Testen",
text: "Jeden Duft mindestens einige Tage tragen. Im Alltag, zu verschiedenen Anlässen und auf der eigenen Haut.",
},
{
number: "03",
title: "Entscheiden",
text: "Full-Size bestellen. CHF 48 werden automatisch angerechnet, sofern der Rabatt noch nicht genutzt wurde.",
},
];
const discoveryComparison = [
{
icon: "×",
title: "Traditioneller Weg",
text: "CHF 180+ für eine Full Size ausgeben, ohne zu wissen, ob sie wirklich passt. Risiko: Fehlkauf, Überforderung oder ein Duft, der im Regal bleibt.",
},
{
icon: "✓",
title: "Discovery Set Weg",
text: "CHF 48 investieren, alle Düfte testen, bewusst entscheiden. Die erste Investition wird einmalig angerechnet der Einstieg bleibt kontrolliert, nachvollziehbar und fair.",
highlight: true,
},
];
function DiscoveryOrderPanel({ onBuy }) {
return ( return (
<div className="discovery-page"> <aside className="discovery-order-panel">
<SharedNavbar variant="light" active="testen" /> <div className="discovery-price-row">
<span>Preis</span>
<main className="discovery-shell"> <strong>CHF 48.</strong>
<div className="discovery-topbar">
<button className="discovery-back-link" type="button" onClick={() => navigate("/")}>
<span className="discovery-back-arrow"></span>
<span>Zurück zur Startseite</span>
</button>
</div> </div>
<section className="discovery-hero" data-reveal-group> <div className="discovery-panel-facts">
<div className="discovery-hero-copy"> {discoveryPanelFacts.map((item) => (
<span className="discovery-kicker" data-reveal="fade"> <div key={item.label}>
DISCOVERY SET <span>{item.label}</span>
</span> <p>{item.value}</p>
<h1 data-reveal="lines">
DER SICHERSTE EINSTIEG
<br />
IN DIE WELT DER NISCHEN-
<br />
DÜFTE
</h1>
<p className="discovery-intro" data-reveal="fade">
6 Düfte × 2ml. Jeden Duft eine Woche tragen. Verstehen, was
wirklich funktioniert. Ohne Risiko.
</p>
<div className="discovery-benefits" data-reveal="fade">
<div className="discovery-benefit">
<span className="discovery-benefit-icon"></span>
<div>
<strong>6 × 2ml Samples aller Signature-Düfte</strong>
<p>
Kalter Beton, Schwarzes Benzin, Verbranntes Chrom, Blasse
Seide, Weisse Asche und Nasser Marmor.
</p>
</div>
</div>
<div className="discovery-benefit">
<span className="discovery-benefit-icon"></span>
<div>
<strong>CHF 48 Gutschein automatisch im Set</strong>
<p>
Nur das erste Discovery Set erstellt den einmaligen Rabatt.
Er wird bei einem späteren Full-Size-Kauf automatisch angerechnet.
</p>
</div>
</div>
<div className="discovery-benefit">
<span className="discovery-benefit-icon"></span>
<div>
<strong>Jedes Sample = ca. 20 Anwendungen</strong>
<p>
Genug, um jeden Duft mehrere Tage im Alltag und in
unterschiedlichen Situationen zu testen.
</p>
</div>
</div>
<div className="discovery-benefit">
<span className="discovery-benefit-icon"></span>
<div>
<strong>Hochwertige Mini-Flacons</strong>
<p>
Sorgfältig zusammengestellt, reduziert gestaltet und als
Teil des atmos Konzepts gedacht.
</p>
</div>
</div>
</div>
<div className="discovery-hero-actions" data-reveal="fade">
<button type="button" className="discovery-primary-btn" onClick={buyDiscoverySet}>
DISCOVERY SET BESTELLEN CHF 48.
</button>
<p>Nur das erste Set erstellt einen einmaligen CHF 48 Full-Size-Rabatt</p>
</div>
</div>
<div className="discovery-mood-grid">
{moodImages.map((image, index) => (
<div className="discovery-mood-tile" key={index}>
<img src={image} alt={`Discovery Mood ${index + 1}`} />
</div> </div>
))} ))}
</div> </div>
</section>
<div className="discovery-panel-actions">
<button type="button" className="discovery-primary-btn" onClick={onBuy}>
Discovery Set bestellen CHF 48.
</button>
<p>Nur das erste Set erstellt einen einmaligen CHF 48 Full-Size-Rabatt.</p>
</div>
</aside>
);
}
function DiscoveryHero({ onBuy }) {
return (
<section className="discovery-hero">
<div className="discovery-hero-stage">
<div className="discovery-hero-copy">
<span className="discovery-kicker">Der Einstieg</span>
<h1>Discovery Set</h1>
<p className="discovery-intro">
6 Düfte × 2ml. Jeden Duft eine Woche tragen. Verstehen, was
wirklich funktioniert. Ohne Risiko. Der sichere Einstieg in die
Welt der Nischendüfte, bevor du dich für eine Full Size entscheidest.
</p>
</div>
<figure className="discovery-hero-visual">
<img
src={DISCOVERY_SET_IMAGE}
alt="Atmos Discovery Set"
loading="eager"
decoding="async"
/>
</figure>
</div>
<DiscoveryOrderPanel onBuy={onBuy} />
</section>
);
}
function DiscoveryStorySection() {
return (
<section className="discovery-story-grid" data-reveal-group>
<div className="discovery-story-copy">
<span className="discovery-label" data-reveal="fade">
Warum Discovery Set
</span>
<h2 data-reveal="lines">Der klügere Einstieg in Nischendüfte.</h2>
<p data-reveal="fade">
Nischen-Parfums sind keine Impulskäufe. Sie brauchen Zeit, um zu
verstehen, wie sie auf deiner Haut funktionieren, wie sie sich im
Alltag entwickeln und ob sie wirklich zu dir passen.
</p>
</div>
<div className="discovery-benefit-panel" data-reveal="fade">
<div className="discovery-benefit-panel-head">
<span>Testlogik</span>
<strong>6 × 2ml</strong>
</div>
{discoveryBenefits.map((benefit) => (
<article className="discovery-benefit" key={benefit.title}>
<span className="discovery-benefit-icon" aria-hidden="true">
</span>
<div>
<strong>{benefit.title}</strong>
<p>{benefit.text}</p>
</div>
</article>
))}
</div>
</section>
);
}
function DiscoveryProcessSection() {
return (
<section className="discovery-process-section" data-reveal-group>
<div className="discovery-section-intro">
<span className="discovery-label" data-reveal="fade">
Ablauf
</span>
<h2 data-reveal="lines">So funktioniert&apos;s</h2>
</div>
<div className="discovery-steps-grid">
{discoverySteps.map((step) => (
<article className="discovery-step-card" key={step.number} data-reveal="fade">
<span className="discovery-step-number">{step.number}</span>
<h3>{step.title}</h3>
<p>{step.text}</p>
</article>
))}
</div>
</section>
);
}
function DiscoveryIncludedSection() {
return (
<section className="discovery-included" data-reveal-group> <section className="discovery-included" data-reveal-group>
<div className="discovery-section-heading"> <div className="discovery-section-heading">
<span className="discovery-label" data-reveal="fade"> <span className="discovery-label" data-reveal="fade">
IM SET ENTHALTEN Im Set enthalten
</span> </span>
<h2 data-reveal="lines">ALLE 6 SIGNATURE-DÜFTE ZUM TESTEN.</h2> <h2 data-reveal="lines">Alle 6 Signature-Düfte zum Testen.</h2>
</div> </div>
<div className="discovery-products-grid"> <div className="discovery-products-grid">
{perfumes.map((perfume) => ( {perfumes.map((perfume) => (
<article className="discovery-product-card" key={perfume.id}> <article className="discovery-product-card" key={perfume.id} data-reveal="fade">
<span className="discovery-product-index">{perfume.id}</span>
<div className="discovery-product-image"> <div className="discovery-product-image">
<img src={perfume.image} alt={perfume.name} /> <img
src={perfume.image}
alt={perfume.name}
loading="lazy"
decoding="async"
/>
</div> </div>
<div className="discovery-product-copy"> <div className="discovery-product-copy">
@ -141,91 +217,81 @@ function DiscoverySetPage() {
))} ))}
</div> </div>
</section> </section>
);
}
<section className="discovery-steps-section"> function DiscoveryComparisonSection() {
<div className="discovery-steps-shell" data-reveal-group> return (
<h2 data-reveal="lines">So funktioniert&apos;s</h2>
<div className="discovery-steps-grid">
<article className="discovery-step-card" data-reveal="fade">
<div className="discovery-step-number">1</div>
<h3>Bestellen</h3>
<p>
Discovery Set für CHF 48 bestellen. Nur dein erstes Set erzeugt
automatisch einen einmaligen Rabatt.
</p>
</article>
<article className="discovery-step-card" data-reveal="fade">
<div className="discovery-step-number">2</div>
<h3>Testen</h3>
<p>
Jeden Duft mindestens einige Tage tragen. Im Alltag, zu
verschiedenen Anlässen und auf der eigenen Haut.
</p>
</article>
<article className="discovery-step-card" data-reveal="fade">
<div className="discovery-step-number">3</div>
<h3>Entscheiden</h3>
<p>
Full-Size bestellen. CHF 48 werden automatisch angerechnet,
sofern der Rabatt noch nicht genutzt wurde.
</p>
</article>
</div>
</div>
</section>
<section className="discovery-comparison-section" data-reveal-group> <section className="discovery-comparison-section" data-reveal-group>
<div className="discovery-section-heading discovery-section-heading--center">
<span className="discovery-label" data-reveal="fade">
WARUM DISCOVERY SET
</span>
<h2 data-reveal="lines">DER KLÜGERE EINSTIEG IN NISCHENDÜFTE.</h2>
<p data-reveal="fade">
Nischen-Parfums sind keine Impulskäufe. Sie brauchen Zeit, um zu
verstehen, wie sie auf deiner Haut funktionieren, wie sie sich im
Alltag entwickeln und ob sie wirklich zu dir passen.
</p>
</div>
<div className="discovery-comparison-grid"> <div className="discovery-comparison-grid">
<div className="discovery-comparison-card" data-reveal="fade"> {discoveryComparison.map((item) => (
<div className="discovery-comparison-head"> <article
<span className="discovery-comparison-icon">×</span> className={`discovery-comparison-card${
<h3>Traditioneller Weg</h3> item.highlight ? " discovery-comparison-card--highlight" : ""
</div> }`}
<p> key={item.title}
CHF 180+ für eine Full Size ausgeben, ohne zu wissen, ob sie
wirklich passt. Risiko: Fehlkauf, Überforderung oder ein Duft,
der im Regal bleibt.
</p>
</div>
<div
className="discovery-comparison-card discovery-comparison-card--highlight"
data-reveal="fade" data-reveal="fade"
> >
<div className="discovery-comparison-head"> <div className="discovery-comparison-head">
<span className="discovery-comparison-icon"></span> <span className="discovery-comparison-icon" aria-hidden="true">
<h3>Discovery Set Weg</h3> {item.icon}
</span>
<h3>{item.title}</h3>
</div> </div>
<p> <p>{item.text}</p>
CHF 48 investieren, alle Düfte testen, bewusst entscheiden. Die </article>
erste Investition wird einmalig angerechnet der Einstieg bleibt ))}
kontrolliert, nachvollziehbar und fair.
</p>
</div>
</div>
<div className="discovery-bottom-cta" data-reveal="fade">
<button type="button" className="discovery-primary-btn" onClick={buyDiscoverySet}>
DISCOVERY SET BESTELLEN CHF 48.
</button>
<p>Kostenloser Versand · 23 Werktage · Einmalige Anrechnung bei Full-Size</p>
</div> </div>
</section> </section>
);
}
function DiscoveryFinalCta({ onBuy }) {
return (
<section className="discovery-final-cta" data-reveal-group>
<div>
<span className="discovery-label" data-reveal="fade">
Discovery Set
</span>
<h2 data-reveal="lines">Der sichere Einstieg.</h2>
<p data-reveal="fade">
Kostenloser Versand · 23 Werktage · Einmalige Anrechnung bei Full-Size
</p>
</div>
<div className="discovery-final-actions" data-reveal="fade">
<button type="button" className="discovery-primary-btn" onClick={onBuy}>
Discovery Set bestellen CHF 48.
</button>
</div>
</section>
);
}
function DiscoverySetPage() {
const navigate = useNavigate();
const { addToCart } = useShop();
const buyDiscoverySet = () =>
addToCart("discovery-set", 1, "Discovery Set added.").catch(() => {});
return (
<div className="discovery-page">
<SharedNavbar variant="hero" active="testen" />
<main className="shell">
<div className="discovery-topbar">
<button className="discovery-back-link" type="button" onClick={() => navigate("/")}>
<span className="discovery-back-arrow" aria-hidden="true" />
<span>Zurück zur Startseite</span>
</button>
</div>
<DiscoveryHero onBuy={buyDiscoverySet} />
<DiscoveryStorySection />
<DiscoveryIncludedSection />
<DiscoveryProcessSection />
<DiscoveryComparisonSection />
<DiscoveryFinalCta onBuy={buyDiscoverySet} />
</main> </main>
</div> </div>
); );

View File

@ -1,152 +1,142 @@
.impressum-page { .impressum-page {
min-height: 100vh; min-height: 100vh;
padding: 0 0 var(--section-y-sm);
color: var(--theme-text); color: var(--theme-text);
padding: 26px 38px 38px;
background: background:
linear-gradient(to right, rgba(0, 0, 0, 0.45), rgba(0, 0, 0, 0.1)), radial-gradient(circle at 86% 8%, rgba(var(--theme-accent-rgb) / 0.12), transparent 28rem),
linear-gradient(to bottom, rgba(0, 0, 0, 0.2), rgba(0, 0, 0, 0.45)); linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
}
.impressum-shell {
background: var(--theme-surface);
border: 1px solid var(--theme-border);
padding: 38px;
} }
.impressum-kicker, .impressum-kicker,
.impressum-label { .impressum-label {
display: block; display: block;
font-size: 10px;
letter-spacing: 0.22em;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-xs);
letter-spacing: 0.22em;
text-transform: uppercase;
} }
.impressum-hero { .impressum-hero {
padding-bottom: 32px; padding: clamp(2rem, 5vw, 5rem) 0 var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); border-bottom: 1px solid var(--theme-border);
} }
.impressum-hero h1 { .impressum-hero h1 {
margin: 14px 0 16px; margin: clamp(0.85rem, 2vw, 1.2rem) 0 clamp(1rem, 2vw, 1.35rem);
font-size: 64px;
line-height: 0.92;
font-weight: 300;
letter-spacing: -0.05em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(3rem, 8vw, 7.2rem);
line-height: 0.9;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
} }
.impressum-intro { .impressum-intro {
max-width: 760px; max-width: var(--text-measure);
margin: 0; margin: 0;
font-size: 17px;
line-height: 1.65;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-lg);
line-height: 1.65;
} }
.impressum-grid { .impressum-grid {
display: grid; display: grid;
grid-template-columns: repeat(2, 1fr); grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 18px; gap: var(--gap-sm);
margin-top: 38px; margin-top: var(--section-y-sm);
}
.impressum-card,
.impressum-note-box {
border: 1px solid var(--theme-border);
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0.008)),
var(--theme-surface-soft);
} }
.impressum-card { .impressum-card {
padding: 24px;
background: var(--theme-bg);
border: 1px solid var(--theme-border);
min-height: 210px; min-height: 210px;
padding: clamp(1.1rem, 2.4vw, 1.8rem);
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
}
.impressum-card:hover {
transform: translateY(-4px);
border-color: rgba(var(--theme-accent-rgb) / 0.42);
box-shadow: var(--theme-shadow-soft);
} }
.impressum-card h2 { .impressum-card h2 {
margin: 14px 0 12px; margin: 0.9rem 0 0.75rem;
font-size: 26px;
line-height: 1.05;
font-weight: 400;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-xl);
line-height: 1.08;
font-weight: 400;
letter-spacing: 0;
} }
.impressum-card p { .impressum-card p {
margin: 0; margin: 0;
font-size: 15px;
line-height: 1.7;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.7;
} }
.impressum-section { .impressum-section {
display: grid; display: grid;
grid-template-columns: minmax(260px, 0.7fr) minmax(0, 1.3fr); grid-template-columns: minmax(14rem, 0.72fr) minmax(0, 1.28fr);
gap: 28px; gap: var(--gap-lg);
align-items: start; align-items: start;
margin-top: 38px; margin-top: var(--section-y-sm);
padding-top: 38px; padding-top: var(--section-y-sm);
border-top: 1px solid var(--theme-border); border-top: 1px solid var(--theme-border);
} }
.impressum-section-heading h2 { .impressum-section-heading h2 {
margin: 10px 0 0; margin: 0.75rem 0 0;
font-size: 38px;
line-height: 0.98;
font-weight: 300;
letter-spacing: -0.04em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(2rem, 4.2vw, 4.5rem);
line-height: 0.96;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.impressum-section-copy p { .impressum-section-copy p {
margin: 0 0 16px; margin: 0 0 1rem;
font-size: 16px;
line-height: 1.75;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.75;
} }
.impressum-note-box { .impressum-note-box {
padding: 22px; padding: clamp(1.1rem, 2.4vw, 1.8rem);
border: 1px solid rgba(255, 106, 0, 0.18); border-color: rgba(var(--theme-accent-rgb) / 0.24);
background: linear-gradient( background:
180deg, linear-gradient(135deg, rgba(var(--theme-accent-rgb) / 0.11), transparent 60%),
rgba(255, 106, 0, 0.08), var(--theme-surface-soft);
rgba(255, 106, 0, 0.03)
);
} }
.impressum-note-box p { .impressum-note-box p {
margin: 0; margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-base);
line-height: 1.65;
} }
@media (max-width: 1100px) { @media (max-width: 900px) {
.impressum-grid, .impressum-grid,
.impressum-section { .impressum-section {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.impressum-hero h1 {
font-size: 48px;
}
} }
@media (max-width: 700px) { @media (max-width: 700px) {
.impressum-page { .impressum-hero {
padding: 18px; padding-top: clamp(1.4rem, 5vw, 2rem);
}
.impressum-shell {
padding: 24px 18px;
}
.impressum-hero h1 {
font-size: 36px;
}
.impressum-section-heading h2 {
font-size: 28px;
}
.impressum-intro,
.impressum-card p,
.impressum-section-copy p {
font-size: 15px;
} }
} }

View File

@ -4,9 +4,9 @@ import "./ImpressumPage.css";
function ImpressumPage() { function ImpressumPage() {
return ( return (
<div className="impressum-page"> <div className="impressum-page">
<SharedNavbar variant="light" /> <SharedNavbar variant="hero" />
<main className="impressum-shell"> <main className="shell">
<section className="impressum-hero" data-reveal-group> <section className="impressum-hero" data-reveal-group>
<span className="impressum-kicker" data-reveal="fade"> <span className="impressum-kicker" data-reveal="fade">
RECHTLICHE ANGABEN RECHTLICHE ANGABEN

View File

@ -1,6 +1,9 @@
.page { .page {
position: relative;
min-height: 100vh; min-height: 100vh;
background: var(--theme-bg); background:
radial-gradient(circle at 82% 12%, rgba(var(--theme-accent-rgb) / 0.13), transparent 28rem),
linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
color: var(--theme-text); color: var(--theme-text);
} }
@ -16,20 +19,37 @@
border: 0; border: 0;
} }
.page main {
padding-bottom: var(--section-y-sm);
}
/* HERO */ /* HERO */
.hero { .hero {
position: relative; position: relative;
width: 100%; width: 100%;
min-height: 100vh; min-height: clamp(680px, 100svh, 980px);
min-height: 100svh;
min-height: 100dvh;
overflow: hidden; overflow: hidden;
display: flex; display: grid;
align-items: center; align-items: center;
isolation: isolate; isolation: isolate;
background: #111; background: #111;
} }
.hero::before {
content: "";
position: absolute;
inset: 0;
z-index: 2;
pointer-events: none;
}
.hero::before {
background:
radial-gradient(circle at 72% 42%, rgba(255, 255, 255, 0.1), transparent 24rem),
linear-gradient(90deg, rgba(0, 0, 0, 0.8) 0%, rgba(0, 0, 0, 0.42) 44%, rgba(0, 0, 0, 0.08) 100%),
linear-gradient(0deg, rgba(0, 0, 0, 0.44), transparent 46%);
}
.hero-media { .hero-media {
position: absolute; position: absolute;
inset: 0; inset: 0;
@ -42,44 +62,50 @@
height: 100%; height: 100%;
display: block; display: block;
object-fit: cover; object-fit: cover;
object-position: center; object-position: 60% center;
will-change: transform; will-change: transform;
} }
.hero .navbar--hero { .hero .navbar--hero {
position: absolute; position: fixed;
top: 22px; top: clamp(0.75rem, 2.1vw, 1.4rem);
left: 0;
right: 0; right: 0;
z-index: 12; left: 0;
z-index: 998;
padding-top: 0; padding-top: 0;
} }
.hero-content { .hero-content {
position: relative; position: relative;
z-index: 6; z-index: 6;
width: min(760px, 100%); width: var(--container-wide);
padding: clamp(6rem, 11vh, 9rem) clamp(1.2rem, 3.4vw, 3rem) margin: 0 auto;
clamp(2.6rem, 7vh, 4rem); padding: clamp(7rem, 14vh, 11rem) 0 clamp(3rem, 8vh, 6rem);
display: flex; display: grid;
flex-direction: column; grid-template-columns: repeat(12, minmax(0, 1fr));
justify-content: center; gap: var(--gap-md);
align-items: center;
} }
.hero-title { .hero-title {
grid-column: 1 / span 7;
max-width: 10.8ch;
margin: 0; margin: 0;
font-size: clamp(2.8rem, 8.5vw, 6.4rem); font-size: clamp(3.2rem, 8.4vw, 8.8rem);
line-height: 0.88; line-height: 0.9;
font-weight: 300; font-weight: 300;
letter-spacing: -0.045em; letter-spacing: 0;
text-transform: uppercase; text-transform: uppercase;
color: #fff; color: #fff;
text-wrap: balance;
} }
.hero-title-line { .hero-title-line {
display: block; display: block;
overflow: hidden; overflow: hidden;
padding-right: 0.12em;
padding-bottom: 0.08em; padding-bottom: 0.08em;
margin-right: -0.12em;
margin-bottom: -0.08em; margin-bottom: -0.08em;
} }
@ -88,63 +114,75 @@
} }
.hero-title-line + .hero-title-line { .hero-title-line + .hero-title-line {
margin-top: 0.1em; margin-top: 0.02em;
} }
.hero-text { .hero-text {
margin-top: 1.25rem; grid-column: 1 / span 5;
max-width: 29rem; max-width: 31rem;
font-size: 0.99rem; margin: 0;
line-height: 1.58; font-size: var(--text-base);
color: rgba(255, 255, 255, 0.86); line-height: 1.62;
color: rgba(255, 255, 255, 0.84);
will-change: transform, opacity; will-change: transform, opacity;
} }
.hero-actions { .hero-actions {
grid-column: 1 / span 5;
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 12px; gap: var(--gap-xs);
margin-top: 1.9rem; margin-top: clamp(0.2rem, 1vw, 0.7rem);
will-change: transform, opacity; will-change: transform, opacity;
} }
.btn { .btn,
.discovery-btn {
min-height: 48px;
border: none; border: none;
border-radius: 999px; border-radius: 999px;
padding: 12px 20px; padding: 0 clamp(1rem, 2vw, 1.35rem);
font-size: 0.9rem; font-size: var(--text-sm);
cursor: pointer; cursor: pointer;
transition: transform 0.24s ease, opacity 0.24s ease; transition:
transform var(--duration-med) var(--ease-out),
opacity var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out);
text-decoration: none; text-decoration: none;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
} }
.hero .btn { .btn:hover,
border-radius: 999px; .discovery-btn:hover {
transform: translateY(-2px);
} }
.btn:hover { .btn:active,
transform: translateY(-1px); .discovery-btn:active {
transform: translateY(0) scale(0.98);
} }
.btn-primary { .btn-primary {
background: #ff6a00; background: var(--theme-accent);
color: #fff; color: #fff;
box-shadow: 0 18px 42px rgba(var(--theme-accent-rgb) / 0.26);
} }
.btn-secondary { .btn-secondary {
background: rgba(255, 255, 255, 0.16); background: rgba(255, 255, 255, 0.13);
color: #fff; color: #fff;
border: 1px solid rgba(255, 255, 255, 0.22); border: 1px solid rgba(255, 255, 255, 0.24);
backdrop-filter: blur(8px); backdrop-filter: blur(12px);
} }
.intro-overlay { .intro-overlay {
position: absolute; position: absolute;
inset: 0; inset: 0;
z-index: 26; z-index: 999;
background: var(--theme-paper); background: var(--theme-paper);
display: grid; display: grid;
place-items: center; place-items: center;
@ -156,7 +194,7 @@
height: 100%; height: 100%;
display: grid; display: grid;
place-items: center; place-items: center;
padding: clamp(1rem, 4vw, 2.2rem); padding: var(--page-x);
} }
.intro-overlay__text-mask { .intro-overlay__text-mask {
@ -179,51 +217,107 @@
/* SECTIONS */ /* SECTIONS */
.section { .section {
padding: 42px 20px 10px; width: var(--container-wide);
margin: 0 auto;
padding: var(--section-y-sm) 0 var(--section-y-xs);
} }
.section-heading { .section-heading {
margin-bottom: 28px; display: grid;
grid-template-columns: repeat(12, minmax(0, 1fr));
gap: var(--gap-md);
align-items: end;
margin-bottom: clamp(1.6rem, 4vw, 4.8rem);
}
.section-heading::after {
content: "01 / Kollektion";
grid-column: 9 / span 3;
align-self: start;
padding-top: 0.3rem;
border-top: 1px solid var(--theme-border);
color: var(--theme-text-muted);
font-size: var(--text-xs);
line-height: 1.4;
letter-spacing: 0.16em;
text-transform: uppercase;
} }
.section-heading h2, .section-heading h2,
.discovery-copy h2 { .discovery-copy h2 {
margin: 0; margin: 0;
font-size: 52px; font-size: clamp(2.6rem, 7vw, 7.4rem);
line-height: 0.95; line-height: 0.92;
font-weight: 300; font-weight: 300;
letter-spacing: -0.04em; letter-spacing: 0;
color: var(--theme-text); color: var(--theme-text);
text-wrap: balance;
}
.section-heading h2 {
grid-column: 1 / span 8;
} }
/* GRID */ /* GRID */
.product-grid { .product-grid {
container-type: inline-size;
display: grid; display: grid;
grid-template-columns: repeat(3, 1fr); grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 18px; gap: var(--gap-sm);
} }
.product-card { .product-card {
position: relative; position: relative;
container-type: inline-size;
isolation: isolate; isolation: isolate;
overflow: hidden; overflow: hidden;
background: var(--theme-surface); min-height: clamp(360px, 36vw, 560px);
border: 1px solid var(--theme-border); display: grid;
border-radius: 0; grid-template-rows: auto minmax(14rem, 1fr) auto;
padding: 18px; padding: clamp(1rem, 2vw, 1.55rem);
min-height: 360px;
display: flex;
flex-direction: column;
justify-content: space-between;
cursor: pointer;
transition: transform 0.15s ease, border-color 0.15s ease;
text-decoration: none;
color: inherit; color: inherit;
text-decoration: none;
background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0)),
var(--theme-surface);
border: 1px solid var(--theme-border);
border-radius: var(--radius-lg);
cursor: pointer;
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
}
.product-card::before {
content: "";
position: absolute;
inset: 0;
z-index: 3;
pointer-events: none;
background:
linear-gradient(180deg, rgba(0, 0, 0, 0.18), transparent 28%),
linear-gradient(0deg, rgba(0, 0, 0, 0.22), transparent 42%);
opacity: 0;
transition: opacity var(--duration-med) var(--ease-out);
}
.product-card:hover,
.product-card:focus-visible {
transform: translateY(-4px);
border-color: rgba(var(--theme-accent-rgb) / 0.48);
box-shadow: var(--theme-shadow-soft);
}
.product-card:hover::before,
.product-card:focus-visible::before {
opacity: 1;
} }
.product-card:focus-visible { .product-card:focus-visible {
outline: 2px solid #ff6a00; outline: 2px solid var(--theme-accent);
outline-offset: 3px; outline-offset: 4px;
} }
.product-hover-fill { .product-hover-fill {
@ -259,14 +353,14 @@
z-index: 1; z-index: 1;
background: linear-gradient( background: linear-gradient(
to bottom, to bottom,
rgba(255, 255, 255, 0.08), rgba(255, 255, 255, 0.06),
rgba(0, 0, 0, 0.18) rgba(0, 0, 0, 0.34)
); );
} }
.product-card:active { .product-card:active {
transform: scale(0.97); transform: translateY(-1px) scale(0.985);
border-color: #ff6a00; border-color: var(--theme-accent);
} }
.product-top { .product-top {
@ -275,20 +369,23 @@
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: flex-start; align-items: flex-start;
gap: 12px; gap: var(--gap-sm);
} }
.product-id { .product-id {
font-size: 18px; font-size: var(--text-sm);
color: var(--theme-text-muted); color: var(--theme-text-muted);
} }
.product-top h3 { .product-top h3 {
max-width: 12ch;
margin: 0; margin: 0;
font-size: 18px; font-size: var(--text-sm);
line-height: 1.15;
font-weight: 400; font-weight: 400;
text-align: right; text-align: right;
letter-spacing: 0.02em; letter-spacing: 0.02em;
text-transform: uppercase;
} }
.product-image-wrap { .product-image-wrap {
@ -297,25 +394,35 @@
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: center; align-items: center;
min-height: 180px;
padding: 20px 0;
width: 100%; width: 100%;
min-height: 0;
padding: clamp(1.2rem, 5vw, 4.8rem) 0;
overflow: hidden; overflow: hidden;
} }
.product-image-wrap::before {
content: "";
position: absolute;
width: min(76%, 26rem);
aspect-ratio: 1;
border-radius: 50%;
background: rgba(var(--theme-accent-rgb) / 0.08);
filter: blur(28px);
transform: translateY(10%);
}
.product-image { .product-image {
position: relative; position: relative;
z-index: 1; z-index: 1;
width: 100%; width: min(92%, 520px);
max-width: 600px;
height: auto; height: auto;
object-fit: contain; object-fit: contain;
border-radius: 0; border-radius: 0;
transition: transform 0.4s ease; transition: transform var(--duration-slow) var(--ease-out);
} }
.product-card:hover .product-image { .product-card:hover .product-image {
transform: scale(1.05); transform: scale(1.045) rotate(-0.6deg);
} }
.product-bottom { .product-bottom {
@ -324,28 +431,47 @@
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: flex-end; align-items: flex-end;
gap: 12px; gap: var(--gap-sm);
} }
.product-bottom p { .product-bottom p {
max-width: 18rem;
margin: 0; margin: 0;
max-width: 170px;
font-size: 15px;
line-height: 1.35;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-sm);
line-height: 1.45;
} }
.arrow { .arrow {
font-size: 26px; display: inline-flex;
color: #ff6a00; align-items: center;
line-height: 1; justify-content: center;
min-width: 2rem;
width: 2rem;
height: 2rem;
color: var(--theme-accent);
font-size: 0;
line-height: 0;
}
.arrow::before {
content: "";
display: block;
width: clamp(1.35rem, 2.2vw, 1.75rem);
height: clamp(1.35rem, 2.2vw, 1.75rem);
background: currentColor;
transform: rotate(180deg);
-webkit-mask: url("/icon-arrow-left.svg") center / contain no-repeat;
mask: url("/icon-arrow-left.svg") center / contain no-repeat;
} }
.product-id, .product-id,
.product-top h3, .product-top h3,
.product-bottom p, .product-bottom p,
.arrow { .arrow {
transition: color 0.25s ease; transition:
color var(--duration-med) var(--ease-out),
transform var(--duration-med) var(--ease-out);
} }
.product-card:hover .product-id, .product-card:hover .product-id,
@ -357,156 +483,232 @@
.product-card:focus-within .product-bottom p, .product-card:focus-within .product-bottom p,
.product-card:focus-within .arrow { .product-card:focus-within .arrow {
color: #fff; color: #fff;
mix-blend-mode: difference;
} }
.product-card:active .product-id, .product-card:hover .arrow,
.product-card:active .product-top h3, .product-card:focus-within .arrow {
.product-card:active .product-bottom p, transform: translateX(0.35rem);
.product-card:active .arrow {
color: #ff6a00;
mix-blend-mode: normal;
transform: scale(1.02);
transition: all 0.1s ease;
} }
/* DISCOVERY */ /* DISCOVERY */
.discovery-section { .discovery-section {
position: relative;
width: var(--container-wide);
min-height: clamp(520px, 62vw, 780px);
margin: var(--section-y-sm) auto 0;
padding: clamp(1.25rem, 4vw, 4rem);
display: grid; display: grid;
grid-template-columns: 600px 1fr; grid-template-columns: minmax(17rem, 0.85fr) minmax(0, 1.35fr);
gap: 28px; gap: var(--gap-lg);
align-items: center; align-items: center;
background: #ff6a00; overflow: hidden;
margin: 5vw 0px 0; background:
border-radius: 0; radial-gradient(circle at 14% 18%, rgba(255, 255, 255, 0.22), transparent 16rem),
padding: 00px 0px 0px 40px; linear-gradient(135deg, #ff6a00, #d84f00);
border-radius: var(--radius-lg);
}
.discovery-section::after {
content: "";
position: absolute;
inset: 1px;
border: 1px solid rgba(255, 255, 255, 0.26);
border-radius: inherit;
pointer-events: none;
}
.discovery-copy {
position: relative;
z-index: 2;
max-width: 36rem;
} }
.discovery-copy h2 { .discovery-copy h2 {
margin: 0;
font-size: 42px;
line-height: 0.95;
font-weight: 300;
letter-spacing: -0.04em;
color: #fff; color: #fff;
font-size: clamp(2.2rem, 5.8vw, 6rem);
} }
.discovery-copy p { .discovery-copy p {
margin-top: 18px; max-width: 29rem;
font-size: 15px; margin: clamp(1rem, 2vw, 1.4rem) 0 0;
line-height: 1.5; color: rgba(255, 255, 255, 0.86);
color: #fff; font-size: var(--text-base);
line-height: 1.62;
} }
.discovery-btn { .discovery-btn {
border: none; margin-top: clamp(1.3rem, 3vw, 2.1rem);
border-radius: 999px; background: #fff;
padding: 12px 18px; color: #d64f00;
font-size: 14px; box-shadow: 0 18px 42px rgba(0, 0, 0, 0.18);
cursor: pointer;
transition: transform 0.2s ease, opacity 0.2s ease;
background: var(--theme-paper);
color: #ff6a00;
text-decoration: none;
display: inline-flex;
align-items: center;
justify-content: center;
}
.discovery-btn:hover {
transform: translateY(-1px);
box-shadow: 0 12px 24px rgba(0, 0, 0, 0.2);
}
.discovery-btn:active {
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.15);
background: rgba(255, 255, 255, 0.8);
} }
.discovery-banner { .discovery-banner {
position: relative; position: relative;
width: 100%; z-index: 1;
max-width: 1300px; width: min(100%, 1080px);
height: 50vh; aspect-ratio: 16 / 10;
border-radius: 0; min-height: 320px;
overflow: hidden;
justify-self: end; justify-self: end;
overflow: hidden;
border-radius: var(--radius-lg);
box-shadow: 0 28px 80px rgba(0, 0, 0, 0.24);
}
.discovery-banner::before {
content: "";
position: absolute;
inset: 0;
z-index: 1;
border: 1px solid rgba(255, 255, 255, 0.24);
border-radius: inherit;
pointer-events: none;
} }
.discovery-banner img { .discovery-banner img {
width: 100%; width: 100%;
height: 100%; height: 100%;
object-fit: cover; object-fit: cover;
object-position: center;
display: block; display: block;
will-change: transform; will-change: transform;
} }
/* RESPONSIVE */ /* RESPONSIVE */
@media (max-width: 900px) { @media (max-width: 1180px) {
.hero-content { .hero-content {
width: min(640px, 100%); grid-template-columns: repeat(8, minmax(0, 1fr));
padding-top: 7rem; }
.hero-title {
grid-column: 1 / span 6;
}
.hero-text,
.hero-actions {
grid-column: 1 / span 4;
}
.section-heading {
grid-template-columns: 1fr;
} }
.hero-title,
.section-heading h2, .section-heading h2,
.discovery-copy h2 { .section-heading::after {
font-size: clamp(2.45rem, 9vw, 3.2rem); grid-column: 1;
} }
.hero-text { .section-heading::after {
font-size: 0.94rem; max-width: 18rem;
} }
.product-grid { .product-grid {
grid-template-columns: repeat(2, 1fr); grid-template-columns: repeat(2, minmax(0, 1fr));
} }
.discovery-section { .discovery-section {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.discovery-banner {
justify-self: stretch;
}
} }
@media (max-width: 640px) { @media (max-width: 760px) {
.hero .navbar--hero { .hero {
top: 14px; min-height: clamp(620px, 91svh, 760px);
}
.hero::before {
background:
linear-gradient(90deg, rgba(0, 0, 0, 0.74), rgba(0, 0, 0, 0.28)),
linear-gradient(0deg, rgba(0, 0, 0, 0.62), transparent 48%);
}
.hero-media__image {
object-position: 64% center;
} }
.hero-content { .hero-content {
padding: 6.2rem 1rem 2.3rem; display: flex;
flex-direction: column;
align-items: flex-start;
justify-content: flex-end;
gap: var(--gap-sm);
padding-top: clamp(6.5rem, 18svh, 8.5rem);
padding-bottom: clamp(2.4rem, 8svh, 4rem);
} }
.hero-title, .hero-title {
.section-heading h2, max-width: 9.6ch;
.discovery-copy h2 { font-size: clamp(3rem, 17.5vw, 5.1rem);
font-size: clamp(2.05rem, 13vw, 2.7rem); }
.hero-text {
max-width: 25rem;
font-size: var(--text-sm);
} }
.hero-actions { .hero-actions {
flex-direction: column; width: min(100%, 22rem);
align-items: flex-start;
width: min(300px, 100%);
} }
.hero-actions .btn { .hero-actions .btn {
width: 100%; width: 100%;
} }
.section {
padding: 34px 12px 10px;
}
.product-grid { .product-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.product-card { .product-card {
min-height: 320px; min-height: clamp(340px, 118vw, 520px);
}
.product-top h3 {
max-width: 11ch;
} }
.discovery-section { .discovery-section {
margin: 12px 12px 0; width: calc(100% - (var(--page-x) * 2));
padding: 28px 20px; min-height: 0;
padding: clamp(1rem, 5vw, 1.5rem);
}
.discovery-banner {
min-height: 0;
aspect-ratio: 1 / 1;
}
}
@media (max-width: 430px) {
.hero-title {
max-width: 9.2ch;
}
.section-heading h2,
.discovery-copy h2 {
font-size: clamp(2.25rem, 13vw, 3.45rem);
}
.product-card {
padding: 1rem;
}
.product-bottom {
align-items: flex-start;
}
}
@container (max-width: 360px) {
.product-bottom {
flex-direction: column;
align-items: flex-start;
}
.arrow {
align-self: flex-end;
} }
} }
@ -516,9 +718,12 @@
.hero-text, .hero-text,
.hero-actions, .hero-actions,
.hero-brand, .hero-brand,
.intro-overlay { .intro-overlay,
transition: none !important; .product-card,
animation: none !important; .product-image,
.discovery-banner img {
transition: none;
animation: none;
will-change: auto;
} }
} }

View File

@ -9,6 +9,8 @@ import { Link } from "react-router";
import { gsap } from "gsap"; import { gsap } from "gsap";
import { ScrollTrigger } from "gsap/ScrollTrigger"; import { ScrollTrigger } from "gsap/ScrollTrigger";
import HeroSection from "../components/landing/HeroSection"; import HeroSection from "../components/landing/HeroSection";
import SharedNavbar from "../components/SharedNavbar";
import { useProductTransition } from "../transitions/ProductTransitionContext";
import perfumes from "../data/perfumes"; import perfumes from "../data/perfumes";
import "../pages/LandingPage.css"; import "../pages/LandingPage.css";
import "../style/navbar.css"; import "../style/navbar.css";
@ -27,6 +29,7 @@ function LandingPage() {
const headlineLineRefs = useRef([]); const headlineLineRefs = useRef([]);
const heroMetaRefs = useRef([]); const heroMetaRefs = useRef([]);
const cardRefs = useRef([]); const cardRefs = useRef([]);
const { startProductTransition } = useProductTransition();
const [introSettings] = useState(() => { const [introSettings] = useState(() => {
if (typeof window === "undefined") { if (typeof window === "undefined") {
@ -375,6 +378,8 @@ function LandingPage() {
return ( return (
<div className="page" ref={pageRef}> <div className="page" ref={pageRef}>
<SharedNavbar variant="hero" active="atmos" />
<HeroSection <HeroSection
heroImageWrapRef={heroImageWrapRef} heroImageWrapRef={heroImageWrapRef}
heroImageRef={heroImageRef} heroImageRef={heroImageRef}
@ -402,6 +407,7 @@ function LandingPage() {
to={`/duft/${item.slug}`} to={`/duft/${item.slug}`}
className="product-card" className="product-card"
key={item.id} key={item.id}
onClick={(event) => startProductTransition(event, item)}
ref={(element) => { ref={(element) => {
cardRefs.current[index] = element; cardRefs.current[index] = element;
}} }}
@ -432,12 +438,19 @@ function LandingPage() {
</div> </div>
<div className="product-image-wrap"> <div className="product-image-wrap">
<img src={item.image} alt={item.name} className="product-image" /> <img
src={item.image}
alt={item.name}
className="product-image"
data-product-transition-source
loading={index < 3 ? "eager" : "lazy"}
decoding="async"
/>
</div> </div>
<div className="product-bottom"> <div className="product-bottom">
<p>{item.text}</p> <p>{item.text}</p>
<span className="arrow">&rarr;</span> <span className="arrow" aria-hidden="true" />
</div> </div>
</Link> </Link>
))} ))}
@ -473,6 +486,7 @@ function LandingPage() {
src="/atmos-discovery-set-thumbnail.png" src="/atmos-discovery-set-thumbnail.png"
alt="Discovery Set" alt="Discovery Set"
loading="lazy" loading="lazy"
decoding="async"
ref={discoveryImageRef} ref={discoveryImageRef}
/> />
</div> </div>

View File

@ -1,127 +1,207 @@
.small-page { .small-page {
min-height: 100vh; min-height: 100vh;
padding: 26px 38px 38px; padding: 0 0 var(--section-y-sm);
background: var(--theme-bg);
color: var(--theme-text); color: var(--theme-text);
} background:
radial-gradient(circle at 84% 8%, rgba(var(--theme-accent-rgb) / 0.13), transparent 28rem),
.small-shell { linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
background: var(--theme-surface);
border: 1px solid var(--theme-border);
padding: 38px;
} }
.small-hero { .small-hero {
max-width: 780px; max-width: 64rem;
padding-bottom: 34px; padding: clamp(2rem, 5vw, 5rem) 0 var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); border-bottom: 1px solid var(--theme-border);
margin-bottom: 28px;
} }
.small-kicker { .small-kicker,
.small-requirement span,
.release-card span {
display: block; display: block;
margin-bottom: 12px; margin-bottom: 0.75rem;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: 10px; font-size: var(--text-xs);
letter-spacing: 0.22em; letter-spacing: 0.22em;
text-transform: uppercase;
} }
.small-hero h1, .small-hero h1,
.small-panel h2 { .small-panel h2 {
margin: 0 0 14px; margin: 0 0 clamp(0.85rem, 2vw, 1.2rem);
color: var(--theme-text);
font-weight: 300;
letter-spacing: 0; letter-spacing: 0;
text-transform: uppercase;
} }
.small-hero h1 { .small-hero h1 {
font-size: clamp(42px, 8vw, 92px); font-size: clamp(3rem, 8.6vw, 9rem);
line-height: 0.92; line-height: 0.88;
text-wrap: balance;
}
.small-panel h2 {
font-size: clamp(2.2rem, 5vw, 5.4rem);
line-height: 0.94;
} }
.small-hero p, .small-hero p,
.small-panel p, .small-panel p,
.release-card p { .release-card p {
max-width: var(--text-measure);
margin: 0; margin: 0;
color: var(--theme-text-muted); color: var(--theme-text-muted);
line-height: 1.55; font-size: var(--text-base);
line-height: 1.65;
} }
.small-panel, .small-panel,
.release-card, .release-card,
.small-error { .small-error {
background: var(--theme-surface-soft);
border: 1px solid var(--theme-border); border: 1px solid var(--theme-border);
padding: 22px; background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0.008)),
var(--theme-surface-soft);
}
.small-panel {
max-width: 68rem;
margin-top: var(--section-y-sm);
padding: clamp(1.25rem, 3vw, 2.2rem);
} }
.small-panel button, .small-panel button,
.release-card button { .release-card button {
min-height: 44px; display: inline-flex;
margin-top: 18px; align-items: center;
border: 1px solid #1f1f1f; justify-content: center;
border-radius: 0; min-height: 48px;
background: #1f1f1f; margin-top: 1.2rem;
padding: 0 1.1rem;
border: 1px solid #111;
border-radius: 999px;
background: #111;
color: #fff; color: #fff;
padding: 0 18px;
cursor: pointer; cursor: pointer;
font-size: var(--text-sm);
text-transform: uppercase;
letter-spacing: 0.1em;
transition:
transform var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out);
}
.small-panel button:hover,
.release-card button:hover,
.small-panel button:focus-visible,
.release-card button:focus-visible {
transform: translateY(-2px);
box-shadow: var(--theme-shadow-soft);
} }
.small-requirements { .small-requirements {
display: grid; display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr)); grid-template-columns: repeat(4, minmax(0, 1fr));
gap: 12px; gap: var(--gap-sm);
margin-top: 18px; margin-top: clamp(1.4rem, 3vw, 2.4rem);
} }
.small-requirement { .small-requirement {
min-height: 7rem;
display: flex; display: flex;
flex-direction: column;
justify-content: space-between; justify-content: space-between;
gap: 12px; gap: var(--gap-xs);
padding: 14px; padding: clamp(1rem, 2vw, 1.4rem);
background: var(--theme-surface-soft);
border: 1px solid var(--theme-border); border: 1px solid var(--theme-border);
background: var(--theme-paper);
} }
.small-requirement span, .small-requirement span {
.release-card span { margin-bottom: 0;
color: var(--theme-text-muted); }
font-size: 10px;
letter-spacing: 0.18em; .small-requirement strong {
text-transform: uppercase; color: var(--theme-text);
font-size: var(--text-base);
font-weight: 400;
line-height: 1.3;
} }
.small-requirement strong.met { .small-requirement strong.met {
color: #ff6a00; color: var(--theme-accent);
}
.small-error {
max-width: 68rem;
margin: var(--gap-sm) 0 0;
padding: clamp(1rem, 2vw, 1.4rem);
border-color: rgba(var(--theme-accent-rgb) / 0.45);
color: var(--theme-text);
} }
.release-grid { .release-grid {
display: grid; display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr)); grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 16px; gap: var(--gap-sm);
margin-top: 22px; margin-top: var(--section-y-sm);
}
.release-card {
min-height: 280px;
display: flex;
flex-direction: column;
justify-content: space-between;
gap: var(--gap-sm);
padding: clamp(1.1rem, 2.4vw, 1.8rem);
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
}
.release-card:hover {
transform: translateY(-4px);
border-color: rgba(var(--theme-accent-rgb) / 0.42);
box-shadow: var(--theme-shadow-soft);
}
.release-card span {
margin-bottom: 0;
} }
.release-card h3 { .release-card h3 {
margin: 10px 0 12px; margin: 0;
color: var(--theme-text);
font-size: var(--text-xl);
line-height: 1.08;
font-weight: 400;
letter-spacing: 0;
} }
.small-error { @media (max-width: 1180px) {
margin: 16px 0 0; .small-requirements,
border-color: #ff6a00; .release-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
} }
@media (max-width: 760px) { @media (max-width: 760px) {
.small-page { .small-hero {
padding: 18px; padding-top: clamp(1.4rem, 5vw, 2rem);
} }
.small-shell { .small-hero h1 {
padding: 24px 18px; font-size: clamp(2.55rem, 13vw, 4.4rem);
} }
.small-requirements, .small-requirements,
.release-grid { .release-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.small-panel button,
.release-card button {
width: 100%;
}
} }

View File

@ -60,9 +60,9 @@ function SmallBatchPage() {
return ( return (
<div className="small-page"> <div className="small-page">
<SharedNavbar variant="light" /> <SharedNavbar variant="hero" />
<main className="small-shell"> <main className="shell">
<section className="small-hero" data-reveal-group> <section className="small-hero" data-reveal-group>
<span className="small-kicker" data-reveal="fade"> <span className="small-kicker" data-reveal="fade">
SMALL BATCH / ARCHIVE / PROTOTYPE SMALL BATCH / ARCHIVE / PROTOTYPE

View File

@ -1,16 +1,10 @@
.support-page { .support-page {
min-height: 100vh; min-height: 100vh;
padding: 0 0 var(--section-y-sm);
color: var(--theme-text); color: var(--theme-text);
padding: 26px 38px 38px;
background: background:
linear-gradient(to right, rgba(0, 0, 0, 0.45), rgba(0, 0, 0, 0.1)), radial-gradient(circle at 86% 8%, rgba(var(--theme-accent-rgb) / 0.13), transparent 28rem),
linear-gradient(to bottom, rgba(0, 0, 0, 0.2), rgba(0, 0, 0, 0.45)); linear-gradient(180deg, var(--theme-bg), color-mix(in srgb, var(--theme-bg) 88%, #000 12%));
}
.support-shell {
background: var(--theme-surface);
border: 1px solid var(--theme-border);
padding: 38px;
} }
.support-kicker, .support-kicker,
@ -18,165 +12,173 @@
.support-panel-label, .support-panel-label,
.support-panel-meta span { .support-panel-meta span {
display: block; display: block;
font-size: 10px;
letter-spacing: 0.22em;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-xs);
letter-spacing: 0.22em;
text-transform: uppercase;
} }
.support-hero { .support-hero {
display: grid; display: grid;
grid-template-columns: minmax(0, 1.35fr) minmax(280px, 0.8fr); grid-template-columns: minmax(0, 1.45fr) minmax(18rem, 0.72fr);
gap: 28px; gap: var(--gap-lg);
align-items: end; align-items: end;
padding-bottom: 36px; padding: clamp(2rem, 5vw, 5rem) 0 var(--section-y-sm);
border-bottom: 1px solid var(--theme-border); border-bottom: 1px solid var(--theme-border);
} }
.support-hero-copy {
min-width: 0;
}
.support-hero-copy h1 { .support-hero-copy h1 {
margin: 14px 0 18px; max-width: 10.8ch;
font-size: 68px; margin: clamp(0.85rem, 2vw, 1.2rem) 0 clamp(1rem, 2vw, 1.35rem);
line-height: 0.92;
font-weight: 300;
letter-spacing: -0.05em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(3rem, 7.4vw, 8.8rem);
line-height: 0.9;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.support-intro { .support-intro {
max-width: 720px; max-width: var(--text-measure);
margin: 0; margin: 0;
font-size: 18px;
line-height: 1.65;
color: var(--theme-text-muted); color: var(--theme-text-muted);
font-size: var(--text-lg);
line-height: 1.65;
} }
.support-hero-panel { .support-hero-panel {
padding: 24px; padding: clamp(1.25rem, 3vw, 2rem);
background: linear-gradient( border: 1px solid rgba(var(--theme-accent-rgb) / 0.2);
180deg, background:
rgba(255, 106, 0, 0.08), linear-gradient(135deg, rgba(var(--theme-accent-rgb) / 0.1), transparent 62%),
rgba(255, 106, 0, 0.03) var(--theme-surface-soft);
);
border: 1px solid rgba(255, 106, 0, 0.18);
} }
.support-hero-panel p { .support-hero-panel p {
margin: 10px 0 0; margin: 0.75rem 0 0;
font-size: 16px;
line-height: 1.6;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-base);
line-height: 1.62;
} }
.support-panel-meta { .support-panel-meta {
display: grid; display: grid;
grid-template-columns: 1fr; gap: var(--gap-sm);
gap: 16px; margin-top: clamp(1.2rem, 2.6vw, 2rem);
margin-top: 24px; padding-top: var(--gap-sm);
padding-top: 20px; border-top: 1px solid rgba(var(--theme-accent-rgb) / 0.2);
border-top: 1px solid rgba(255, 106, 0, 0.14);
} }
.support-panel-meta p { .support-panel-meta p {
margin: 8px 0 0; margin: 0.45rem 0 0;
font-size: 14px;
line-height: 1.55;
color: var(--theme-text); color: var(--theme-text);
font-size: var(--text-sm);
line-height: 1.55;
}
.support-quick-grid,
.support-info-grid,
.support-faq-grid {
display: grid;
gap: var(--gap-sm);
} }
.support-quick-grid { .support-quick-grid {
grid-template-columns: repeat(3, minmax(0, 1fr));
margin-top: var(--section-y-sm);
}
.support-section,
.support-faq-section {
padding-top: var(--section-y-sm);
}
.support-section--split,
.support-info-grid {
display: grid; display: grid;
grid-template-columns: repeat(3, 1fr); grid-template-columns: minmax(16rem, 0.72fr) minmax(0, 1.28fr);
gap: 18px; gap: var(--gap-lg);
margin-top: 38px;
}
.support-quick-card {
padding: 24px;
background: var(--theme-bg);
border: 1px solid var(--theme-border);
min-height: 240px;
}
.support-quick-card h3 {
margin: 14px 0 12px;
font-size: 24px;
line-height: 1.08;
font-weight: 400;
color: var(--theme-text);
}
.support-quick-card p {
margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text-muted);
}
.support-section {
padding-top: 38px;
}
.support-section--split {
display: grid;
grid-template-columns: minmax(260px, 0.7fr) minmax(0, 1.3fr);
gap: 28px;
align-items: start; align-items: start;
} }
.support-info-grid {
margin-top: var(--section-y-sm);
}
.support-section-heading h2, .support-section-heading h2,
.support-bottom-copy h2 { .support-bottom-copy h2 {
margin: 10px 0 0; margin: 0.75rem 0 0;
font-size: 42px;
line-height: 0.98;
font-weight: 300;
letter-spacing: -0.04em;
color: var(--theme-text); color: var(--theme-text);
font-size: clamp(2.15rem, 5.2vw, 6rem);
line-height: 0.94;
font-weight: 300;
letter-spacing: 0;
text-transform: uppercase;
text-wrap: balance;
} }
.support-section-copy { .support-section-copy {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 18px; gap: var(--gap-sm);
} }
.support-section-copy p, .support-section-copy p,
.support-bottom-copy p { .support-bottom-copy p,
.support-quick-card p,
.support-faq-card p,
.support-info-box p,
.support-list li {
margin: 0; margin: 0;
font-size: 16px; color: var(--theme-text-muted);
font-size: var(--text-base);
line-height: 1.7; line-height: 1.7;
color: var(--theme-text-muted);
}
.support-info-grid {
display: grid;
grid-template-columns: minmax(0, 1fr) minmax(320px, 0.9fr);
gap: 18px;
margin-top: 38px;
} }
.support-quick-card,
.support-faq-card,
.support-info-box { .support-info-box {
padding: 24px; min-height: 100%;
padding: clamp(1.1rem, 2.4vw, 1.8rem);
border: 1px solid var(--theme-border); border: 1px solid var(--theme-border);
background: var(--theme-surface-soft); background:
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0.008)),
var(--theme-surface-soft);
transition:
transform var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
} }
.support-quick-card:hover,
.support-faq-card:hover,
.support-info-box:hover {
transform: translateY(-4px);
border-color: rgba(var(--theme-accent-rgb) / 0.42);
box-shadow: var(--theme-shadow-soft);
}
.support-quick-card h3,
.support-faq-card h3,
.support-info-box h3 { .support-info-box h3 {
margin: 14px 0 12px; margin: 0.9rem 0 0.75rem;
font-size: 28px; color: var(--theme-text);
line-height: 1.02; font-size: var(--text-xl);
line-height: 1.08;
font-weight: 400; font-weight: 400;
color: #fff; letter-spacing: 0;
}
.support-info-box p {
margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text-muted);
} }
.support-info-box--dark { .support-info-box--dark {
background: #1f1f1f; border-color: rgba(255, 255, 255, 0.16);
border-color: var(--theme-text); background:
radial-gradient(circle at 100% 0%, rgba(var(--theme-accent-rgb) / 0.18), transparent 18rem),
#171717;
} }
.support-info-box--dark .support-label, .support-info-box--dark .support-label,
@ -184,80 +186,61 @@
color: rgba(255, 255, 255, 0.78); color: rgba(255, 255, 255, 0.78);
} }
.support-list { .support-info-box--dark h3 {
margin: 14px 0 0; color: #fff;
padding-left: 18px;
display: grid;
gap: 10px;
} }
.support-list li { .support-list {
font-size: 15px; display: grid;
line-height: 1.6; gap: 0.75rem;
color: var(--theme-text); margin: 1rem 0 0;
padding-left: 1.1rem;
} }
.support-mail-btn { .support-mail-btn {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
margin-top: 18px; min-height: 48px;
text-decoration: none; margin-top: 1.15rem;
padding: 12px 18px; padding: 0 1.1rem;
border-radius: 999px; border-radius: 999px;
background: #ff6a00; background: var(--theme-accent);
color: #fff; color: #fff;
font-size: 14px; font-size: var(--text-sm);
transition: transform 0.2s ease, opacity 0.2s ease, box-shadow 0.2s ease; text-decoration: none;
transition:
transform var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out);
} }
.support-mail-btn:hover { .support-mail-btn:hover,
transform: translateY(-1px); .support-mail-btn:focus-visible {
transform: translateY(-2px);
box-shadow: var(--theme-shadow-soft);
} }
.support-faq-section { .support-faq-section {
margin-top: 38px; margin-top: var(--section-y-sm);
padding-top: 38px;
border-top: 1px solid var(--theme-border); border-top: 1px solid var(--theme-border);
} }
.support-faq-grid { .support-faq-grid {
display: grid; grid-template-columns: repeat(2, minmax(0, 1fr));
grid-template-columns: repeat(2, 1fr); margin-top: clamp(1.6rem, 4vw, 3rem);
gap: 18px;
margin-top: 28px;
}
.support-faq-card {
padding: 24px;
border: 1px solid var(--theme-border);
background: var(--theme-bg);
min-height: 210px;
}
.support-faq-card h3 {
margin: 0 0 12px;
font-size: 22px;
line-height: 1.08;
font-weight: 400;
color: var(--theme-text);
}
.support-faq-card p {
margin: 0;
font-size: 15px;
line-height: 1.65;
color: var(--theme-text-muted);
} }
.support-bottom-cta { .support-bottom-cta {
margin-top: 38px;
padding: 38px;
background: #ff6a00;
display: grid; display: grid;
grid-template-columns: minmax(0, 1.2fr) auto; grid-template-columns: minmax(0, 1fr) auto;
gap: 24px; gap: var(--gap-lg);
align-items: end; align-items: end;
margin-top: var(--section-y-sm);
padding: clamp(1.5rem, 4vw, 3.5rem);
overflow: hidden;
background:
radial-gradient(circle at 92% 0%, rgba(255, 255, 255, 0.22), transparent 20rem),
var(--theme-accent);
} }
.support-bottom-copy .support-label, .support-bottom-copy .support-label,
@ -266,48 +249,54 @@
color: #fff; color: #fff;
} }
.support-bottom-copy .support-label {
opacity: 0.85;
}
.support-bottom-copy p { .support-bottom-copy p {
margin-top: 16px; max-width: 48rem;
max-width: 700px; margin-top: 1rem;
} }
.support-bottom-actions { .support-bottom-actions {
display: flex; display: flex;
gap: 12px;
flex-wrap: wrap; flex-wrap: wrap;
gap: var(--gap-xs);
justify-content: flex-end;
} }
.support-btn { .support-btn {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
text-decoration: none; min-height: 48px;
padding: 12px 18px; padding: 0 1.1rem;
font-size: 14px; border: 1px solid transparent;
border-radius: 999px; border-radius: 999px;
transition: transform 0.2s ease, opacity 0.2s ease, box-shadow 0.2s ease; color: inherit;
font-size: var(--text-sm);
text-decoration: none;
transition:
transform var(--duration-med) var(--ease-out),
box-shadow var(--duration-med) var(--ease-out),
background-color var(--duration-med) var(--ease-out);
} }
.support-btn:hover { .support-btn:hover,
transform: translateY(-1px); .support-btn:focus-visible {
transform: translateY(-2px);
box-shadow: var(--theme-shadow-soft);
} }
.support-btn--primary { .support-btn--primary {
background: var(--theme-paper); background: #fff;
color: #ff6a00; color: var(--theme-accent);
} }
.support-btn--secondary { .support-btn--secondary {
border-color: rgba(255, 255, 255, 0.22);
background: rgba(255, 255, 255, 0.14); background: rgba(255, 255, 255, 0.14);
color: #fff; color: #fff;
backdrop-filter: blur(8px); backdrop-filter: blur(8px);
} }
@media (max-width: 1100px) { @media (max-width: 1180px) {
.support-hero, .support-hero,
.support-section--split, .support-section--split,
.support-info-grid, .support-info-grid,
@ -315,44 +304,32 @@
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.support-quick-grid,
.support-faq-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
}
@media (max-width: 760px) {
.support-hero {
padding-top: clamp(1.4rem, 5vw, 2rem);
}
.support-hero-copy h1 { .support-hero-copy h1 {
font-size: 52px; font-size: clamp(2.55rem, 13vw, 4.4rem);
} }
.support-quick-grid, .support-quick-grid,
.support-faq-grid { .support-faq-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
} }
.support-bottom-actions {
display: grid;
justify-content: stretch;
} }
@media (max-width: 700px) { .support-btn {
.support-page { width: 100%;
padding: 18px;
}
.support-shell {
padding: 24px 18px;
}
.support-hero-copy h1 {
font-size: 38px;
}
.support-section-heading h2,
.support-bottom-copy h2 {
font-size: 30px;
}
.support-intro,
.support-section-copy p,
.support-bottom-copy p,
.support-quick-card p,
.support-faq-card p {
font-size: 15px;
}
.support-bottom-cta {
padding: 26px 20px;
} }
} }

View File

@ -5,9 +5,9 @@ import "./SupportPage.css";
function SupportPage() { function SupportPage() {
return ( return (
<div className="support-page"> <div className="support-page">
<SharedNavbar variant="light" /> <SharedNavbar variant="hero" />
<main className="support-shell"> <main className="shell">
<section className="support-hero" data-reveal-group> <section className="support-hero" data-reveal-group>
<div className="support-hero-copy"> <div className="support-hero-copy">
<span className="support-kicker" data-reveal="fade"> <span className="support-kicker" data-reveal="fade">

View File

@ -1,39 +1,86 @@
/* --- Shared Navbar Start --- */ /* --- Shared Navbar Start --- */
.navbar { .navbar {
position: relative; position: sticky;
z-index: 20; top: clamp(0.75rem, 2vw, 1.25rem);
z-index: 998;
display: flex; display: flex;
justify-content: center; justify-content: center;
width: 100%;
padding-inline: var(--page-x);
} }
.nav-pill { .nav-pill {
display: flex; display: flex;
gap: 10px; align-items: center;
padding: 8px 10px; justify-content: center;
gap: clamp(0.25rem, 0.8vw, 0.65rem);
width: fit-content;
max-width: 100%;
min-height: clamp(3rem, 5.4vw, 3.5rem);
padding: clamp(0.32rem, 0.8vw, 0.55rem);
border: 1px solid transparent;
border-radius: 999px; border-radius: 999px;
backdrop-filter: blur(10px); backdrop-filter: blur(18px) saturate(1.2);
-webkit-backdrop-filter: blur(10px); -webkit-backdrop-filter: blur(18px) saturate(1.2);
box-shadow: 0 16px 50px rgba(0, 0, 0, 0.16);
} }
.nav-link { .nav-link {
position: relative;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
text-decoration: none; min-height: 42px;
font-size: 13px; min-width: 42px;
padding: 8px 14px; padding: 0 clamp(0.72rem, 1.5vw, 1rem);
border-radius: 999px; border-radius: 999px;
transition: 0.2s ease; color: inherit;
font-size: var(--text-xs);
line-height: 1;
text-decoration: none;
white-space: nowrap;
transition:
background-color var(--duration-med) var(--ease-out),
color var(--duration-med) var(--ease-out),
opacity var(--duration-med) var(--ease-out),
transform var(--duration-med) var(--ease-out);
}
.nav-link::after {
content: "";
position: absolute;
right: 0.8rem;
bottom: 0.45rem;
left: 0.8rem;
height: 1px;
background: currentColor;
opacity: 0;
transform: scaleX(0.35);
transform-origin: center;
transition:
opacity var(--duration-med) var(--ease-out),
transform var(--duration-med) var(--ease-out);
}
.nav-link:hover::after,
.nav-link.active::after {
opacity: 0.5;
transform: scaleX(1);
} }
.nav-link--brand { .nav-link--brand {
padding: 8px 12px; padding-inline: clamp(0.75rem, 1.4vw, 1rem);
}
.nav-link--brand::after,
.nav-theme-switch::after {
display: none;
} }
.nav-brand-logo { .nav-brand-logo {
display: block; display: block;
width: clamp(56px, 5.4vw, 78px); width: clamp(58px, 5.2vw, 82px);
height: auto; height: auto;
} }
@ -44,28 +91,32 @@
} }
.nav-theme-switch { .nav-theme-switch {
min-width: auto; min-width: 46px;
padding: 8px; padding-inline: 0.55rem;
} }
.nav-theme-switch__track { .nav-theme-switch__track {
position: relative; position: relative;
width: 38px; width: 40px;
height: 20px; height: 22px;
border-radius: 999px;
border: 1px solid;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
padding: 2px; padding: 2px;
transition: background-color 0.2s ease, border-color 0.2s ease; border: 1px solid;
border-radius: 999px;
transition:
background-color var(--duration-med) var(--ease-out),
border-color var(--duration-med) var(--ease-out);
} }
.nav-theme-switch__thumb { .nav-theme-switch__thumb {
width: 14px; width: 16px;
height: 14px; height: 16px;
border-radius: 50%; border-radius: 50%;
transform: translateX(0); transform: translateX(0);
transition: transform 0.2s ease, background-color 0.2s ease; transition:
transform var(--duration-med) var(--ease-snap),
background-color var(--duration-med) var(--ease-out);
} }
.nav-theme-switch.is-light .nav-theme-switch__thumb { .nav-theme-switch.is-light .nav-theme-switch__thumb {
@ -74,15 +125,21 @@
/* Hero variant */ /* Hero variant */
.navbar--hero { .navbar--hero {
padding-top: 22px; position: fixed;
top: clamp(0.75rem, 2.1vw, 1.4rem);
left: 0;
right: 0;
z-index: 998;
padding-top: 0;
} }
.navbar--hero .nav-pill { .navbar--hero .nav-pill {
background: rgba(255, 255, 255, 0.15); background: rgba(15, 15, 15, 0.58);
border-color: rgba(255, 255, 255, 0.22);
} }
.navbar--hero .nav-link { .navbar--hero .nav-link {
color: rgba(255, 255, 255, 0.88); color: rgba(255, 255, 255, 0.9);
} }
.navbar--hero .nav-button { .navbar--hero .nav-button {
@ -100,17 +157,18 @@
.navbar--hero .nav-link:hover, .navbar--hero .nav-link:hover,
.navbar--hero .nav-link.active { .navbar--hero .nav-link.active {
background: rgba(255, 255, 255, 0.22); background: rgba(255, 255, 255, 0.16);
} }
/* Detail page variant */ /* Detail page variant */
.navbar--light { .navbar--light {
margin-bottom: 18px; margin-bottom: clamp(1rem, 2.5vw, 1.8rem);
padding-top: clamp(0.35rem, 1vw, 0.7rem);
} }
.navbar--light .nav-pill { .navbar--light .nav-pill {
background: rgba(255, 255, 255, 0.88); background: color-mix(in srgb, var(--theme-paper) 86%, transparent);
border: 1px solid var(--theme-border); border-color: var(--theme-border);
} }
.navbar--light .nav-link { .navbar--light .nav-link {
@ -122,7 +180,7 @@
} }
.navbar--light .nav-theme-switch__track { .navbar--light .nav-theme-switch__track {
border-color: rgba(38, 38, 38, 0.24); border-color: var(--theme-border-strong);
background: rgba(38, 38, 38, 0.08); background: rgba(38, 38, 38, 0.08);
} }
@ -137,19 +195,31 @@
/* --- Shared Navbar End --- */ /* --- Shared Navbar End --- */
@media (max-width: 640px) { @media (max-width: 700px) {
.navbar {
padding-inline: clamp(0.75rem, 4vw, 1rem);
}
.nav-pill { .nav-pill {
gap: 4px; justify-content: space-between;
padding: 6px; width: 100%;
overflow-x: auto;
scrollbar-width: none;
}
.nav-pill::-webkit-scrollbar {
display: none;
} }
.nav-link { .nav-link {
padding: 8px 10px; min-height: 40px;
font-size: 12px; min-width: 40px;
padding-inline: 0.68rem;
font-size: 0.75rem;
} }
.nav-link--brand { .nav-link--brand {
padding: 8px 10px; padding-inline: 0.72rem;
} }
.nav-brand-logo { .nav-brand-logo {
@ -157,8 +227,16 @@
} }
.nav-theme-switch { .nav-theme-switch {
min-width: auto; padding-inline: 0.45rem;
padding: 8px;
} }
} }
@media (max-width: 390px) {
.nav-link {
padding-inline: 0.58rem;
}
.nav-link--brand {
padding-inline: 0.62rem;
}
}

View File

@ -1,17 +1,7 @@
import { createContext, useContext } from "react"; import { ThemeContext } from "./ThemeContextBase";
const ThemeContext = createContext(null);
function ThemeProvider({ value, children }) { function ThemeProvider({ value, children }) {
return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>; return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>;
} }
function useTheme() { export { ThemeProvider };
const context = useContext(ThemeContext);
if (!context) {
throw new Error("useTheme must be used within ThemeProvider.");
}
return context;
}
export { ThemeProvider, useTheme };

View File

@ -0,0 +1,5 @@
import { createContext } from "react";
const ThemeContext = createContext(null);
export { ThemeContext };

View File

@ -0,0 +1,12 @@
import { useContext } from "react";
import { ThemeContext } from "./ThemeContextBase";
function useTheme() {
const context = useContext(ThemeContext);
if (!context) {
throw new Error("useTheme must be used within ThemeProvider.");
}
return context;
}
export { useTheme };

View File

@ -0,0 +1,9 @@
import { createContext, useContext } from "react";
export const ProductTransitionContext = createContext({
activeSlug: null,
phase: "idle",
startProductTransition: () => false,
});
export const useProductTransition = () => useContext(ProductTransitionContext);