/* ludoc.net — DA commune (thème sombre VHS) + styles portfolio.
   Tokens & loader repris du système partagé de setup.ludoc.net. */
:root {
  --bg: #0c0d10;
  --bg-2: #121419;
  --card: #16181f;
  --card-hover: #1c1f28;
  --text: #f4f5f7;
  --muted: #9aa0ac;
  --line: rgba(255, 255, 255, 0.08);
  --accent: #6c7bff;
  --accent-2: #ff5f9e;
  --hover-line: rgba(255, 255, 255, 0.15);
  --radius: 18px;
  --shadow: 0 18px 50px rgba(0, 0, 0, 0.45);
}
* { box-sizing: border-box; }
.sr-only { position: absolute; width: 1px; height: 1px; padding: 0; margin: -1px; overflow: hidden; clip: rect(0,0,0,0); white-space: nowrap; border: 0; }

/* ============ LOADER VHS ============ */
html.skip-loader #loader { display: none !important; }
#loader { position: fixed; inset: 0; z-index: 9999; display: grid; place-items: center; background: #07080a; overflow: hidden; animation: loaderOut .55s ease 2s forwards, vhsFlicker .5s steps(2) infinite; }
#loader::after { content: ""; position: absolute; inset: 0; pointer-events: none; background: radial-gradient(120% 120% at 50% 50%, transparent 58%, rgba(0,0,0,.85) 100%); }
.vhs { position: relative; width: min(200px, 46vw); filter: saturate(1.15) contrast(1.04); animation: vhsIn .6s ease forwards, vhsJitter .9s steps(4) infinite; }
.vhs-logo { position: relative; }
.vhs-logo img { display: block; width: 100%; height: auto; filter: brightness(.95); }
.ch { position: absolute; inset: 0; pointer-events: none; -webkit-mask: var(--logo) center / contain no-repeat; mask: var(--logo) center / contain no-repeat; mix-blend-mode: screen; }
.ch-r { background: #ff0033; animation: chromaR 1.6s ease-out forwards; }
.ch-b { background: #00ccff; animation: chromaB 1.6s ease-out forwards; }
.vhs-lines { position: absolute; inset: 0; pointer-events: none; -webkit-mask: var(--logo) center / contain no-repeat; mask: var(--logo) center / contain no-repeat; background: repeating-linear-gradient(to bottom, rgba(0,0,0,0) 0px, rgba(0,0,0,0) 3px, rgba(0,0,0,.18) 4px, rgba(0,0,0,.18) 5px); mix-blend-mode: multiply; opacity: .4; }
.vhs-noise { position: absolute; inset: 0; pointer-events: none; opacity: .1; -webkit-mask: var(--logo) center / contain no-repeat; mask: var(--logo) center / contain no-repeat; background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='140' height='140'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.85' numOctaves='2' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)'/%3E%3C/svg%3E"); background-size: 140px 140px; mix-blend-mode: screen; animation: vhsNoise .18s steps(4) infinite; }
.vhs-track { position: absolute; inset: 0; pointer-events: none; -webkit-mask: var(--logo) center / contain no-repeat; mask: var(--logo) center / contain no-repeat; background: linear-gradient(180deg, transparent 42%, rgba(255,255,255,.06) 47%, rgba(255,255,255,.14) 50%, rgba(255,255,255,.06) 53%, transparent 58%); background-size: 100% 300%; background-position: 0 150%; mix-blend-mode: screen; animation: vhsTrack 3.4s linear infinite; }
.vhs-osd { position: absolute; pointer-events: none; z-index: 2; font-family: "Courier New", monospace; font-weight: 700; font-size: clamp(.66rem, 1.9vw, .82rem); letter-spacing: .1em; color: #f4f5f7; text-shadow: 0 0 5px rgba(0,0,0,.9); opacity: 0; animation: osdIn .5s ease .35s forwards; }
.vhs-osd-tl { top: -30px; left: -10px; color: #ff6b6b; }
.vhs-osd-tr { top: -30px; right: -10px; }
.vhs-osd-br { bottom: -30px; right: -10px; }
.rec { display: inline-flex; align-items: center; gap: 7px; }
.rec i { width: 8px; height: 8px; border-radius: 50%; background: #ff3b3b; box-shadow: 0 0 6px #ff3b3b; animation: recBlink 1.6s steps(2) infinite; }
@keyframes vhsIn { from { opacity: 0; transform: scale(.96); } to { opacity: 1; transform: none; } }
@keyframes vhsJitter { 0% { transform: translateX(0); } 50% { transform: translateX(-.5px); } 100% { transform: translateX(.5px); } }
@keyframes vhsFlicker { 0% { filter: brightness(1); } 100% { filter: brightness(1.02); } }
@keyframes vhsNoise { 0% { background-position: 0 0; } 25% { background-position: -40px 30px; } 50% { background-position: 30px -50px; } 75% { background-position: -20px 60px; } 100% { background-position: 50px 20px; } }
@keyframes vhsTrack { from { background-position: 0 150%; } to { background-position: 0 -150%; } }
@keyframes chromaR { 0% { transform: translate(-5px, 1px); opacity: .6; } 60% { transform: translate(-2px, 0); opacity: .3; } 100% { transform: translate(0, 0); opacity: 0; } }
@keyframes chromaB { 0% { transform: translate(5px, -1px); opacity: .6; } 60% { transform: translate(2px, 0); opacity: .3; } 100% { transform: translate(0, 0); opacity: 0; } }
@keyframes osdIn { to { opacity: .7; } }
@keyframes recBlink { 0% { opacity: 1; } 100% { opacity: .15; } }
@keyframes loaderOut { to { opacity: 0; visibility: hidden; } }

/* ============ BASE ============ */
html { scroll-behavior: smooth; }
body { margin: 0; font-family: "Inter", system-ui, -apple-system, sans-serif; color: var(--text); background: radial-gradient(circle at 50% -10%, #1a1d27 0%, var(--bg) 55%); min-height: 100vh; overflow-x: hidden; -webkit-font-smoothing: antialiased; }
.bg-orbs { position: fixed; inset: 0; z-index: 0; pointer-events: none; overflow: hidden; }
.orb { position: absolute; border-radius: 50%; filter: blur(110px); opacity: 0.4; }
.orb-1 { width: 620px; height: 620px; background: var(--accent); top: -160px; left: -140px; }
.orb-2 { width: 560px; height: 560px; background: var(--accent-2); top: 25%; right: -160px; opacity: 0.32; }
.orb-3 { width: 520px; height: 520px; background: #28d8b0; bottom: -180px; left: 25%; opacity: 0.22; }

/* ============ HERO ============ */
.hero { position: relative; z-index: 1; text-align: center; padding: 64px 20px 18px; }

/* Avatar : halo dégradé animé + anneau de verre (cohérent avec le linktree) */
.hero__avatar { position: relative; width: 132px; height: 132px; margin: 0 auto 20px; display: grid; place-items: center; }
.hero__halo { position: absolute; inset: -8px; border-radius: 50%; background: conic-gradient(from 0deg, var(--accent), var(--accent-2), #28d8b0, var(--accent)); filter: blur(9px); opacity: .5; z-index: 0; animation: heroHalo 9s linear infinite; }
@keyframes heroHalo { to { transform: rotate(360deg); } }
.hero__ring { position: relative; z-index: 1; width: 100%; height: 100%; border-radius: 50%; padding: 4px; background: linear-gradient(160deg, rgba(255,255,255,.2), rgba(255,255,255,.02)); display: grid; place-items: center; box-shadow: var(--shadow); }
.hero__logo { width: 100%; height: 100%; border-radius: 50%; object-fit: cover; background: #0e0f13; display: block; transition: transform .35s ease, filter .35s ease; }
.hero__avatar:hover .hero__logo { transform: scale(1.05); filter: drop-shadow(-1.5px 0 #ff0033) drop-shadow(1.5px 0 #00ccff) contrast(1.08) saturate(1.15); }

/* Eyebrow fileté */
.hero__eyebrow { display: inline-flex; align-items: center; gap: 13px; margin: 0 0 12px; color: var(--muted); font-size: .74rem; font-weight: 600; text-transform: uppercase; letter-spacing: .22em; }
.hero__eyebrow::before, .hero__eyebrow::after { content: ""; width: 28px; height: 1px; }
.hero__eyebrow::before { background: linear-gradient(90deg, transparent, var(--accent)); }
.hero__eyebrow::after { background: linear-gradient(90deg, var(--accent-2), transparent); }

.hero__name { margin: 0 0 4px; font-family: "Montserrat", "Inter", sans-serif; font-weight: 800; font-size: clamp(2.4rem, 8vw, 3.8rem); letter-spacing: .05em; line-height: 1; background: linear-gradient(90deg, #fff 0%, #cfd5ff 52%, var(--accent) 100%); -webkit-background-clip: text; background-clip: text; -webkit-text-fill-color: transparent; color: transparent; }

/* Parcours : ligne sobre, séparateurs fins (pas de pastilles ni de points colorés) */
.hero__credits { list-style: none; display: flex; flex-wrap: wrap; justify-content: center; align-items: center; gap: 0; margin: 16px auto 0; padding: 0; max-width: 580px; }
.hero__credits li { position: relative; color: var(--muted); font-size: .82rem; font-weight: 500; letter-spacing: .04em; padding: 2px 18px; }
.hero__credits li + li::before { content: ""; position: absolute; left: 0; top: 50%; transform: translateY(-50%); width: 1px; height: 12px; background: var(--line); }
@media (max-width: 480px) { .hero__credits li + li::before { display: none; } }

/* Entrée douce des éléments du hero */
.hero__avatar, .hero__eyebrow, .hero__name, .hero__credits { opacity: 0; transform: translateY(12px); animation: heroIn .6s cubic-bezier(.16,.7,.18,1) forwards; }
.hero__eyebrow { animation-delay: .06s; }
.hero__name { animation-delay: .12s; }
.hero__credits { animation-delay: .18s; }
@keyframes heroIn { to { opacity: 1; transform: none; } }
@media (prefers-reduced-motion: reduce) {
  .hero__halo { animation: none; }
  .hero__avatar, .hero__eyebrow, .hero__name, .hero__credits { opacity: 1; transform: none; animation: none; }
}

/* ============ LAYOUT ============ */
.wrap { position: relative; z-index: 1; max-width: 1180px; margin: 0 auto; padding: 18px 20px 40px; }

.lg-defs { position: absolute; width: 0; height: 0; overflow: hidden; pointer-events: none; }

/* ============ GRILLE PORTFOLIO ============ */
.grid { display: grid; gap: 20px; grid-template-columns: repeat(auto-fill, minmax(320px, 1fr)); margin-top: 8px; }
.card { position: relative; display: flex; flex-direction: column; background: var(--card); border: 1px solid var(--line); border-radius: var(--radius); overflow: hidden; cursor: pointer; transform: translateZ(0); -webkit-mask-image: -webkit-radial-gradient(white, black); transition: background .25s ease, border-color .25s ease, box-shadow .25s ease; }
.card:hover, .card:focus-visible { background: var(--card-hover); border-color: var(--hover-line); box-shadow: var(--shadow); outline: none; }
.card__media { position: relative; aspect-ratio: 16 / 9; background-image: var(--thumb); background-size: cover; background-position: center; transition: transform .5s cubic-bezier(.16,.7,.18,1); }
.card:hover .card__media { transform: scale(1.04); }
.card__media::after { content: ""; position: absolute; inset: 0; z-index: 2; background: linear-gradient(180deg, transparent 45%, rgba(8,9,12,.55) 100%); }
/* Encart catégorie posé sur la miniature (en haut à gauche) — verre dépoli léger */
.card__badge { position: absolute; z-index: 3; top: 12px; left: 12px; font-family: "Inter", sans-serif; font-size: .68rem; font-weight: 700; letter-spacing: .08em; text-transform: uppercase; color: #fff; padding: 5px 11px; border-radius: 999px; background: rgba(12,13,16,.5); border: 1px solid rgba(255,255,255,.18); backdrop-filter: blur(8px) saturate(1.4); -webkit-backdrop-filter: blur(8px) saturate(1.4); box-shadow: inset 0 1px 0 rgba(255,255,255,.18), 0 4px 12px rgba(0,0,0,.3); }
.card__badge::before { content: ""; display: inline-block; width: 6px; height: 6px; margin-right: 6px; vertical-align: middle; border-radius: 50%; background: linear-gradient(120deg, var(--accent), var(--accent-2)); }
/* Grain film très léger par-dessus la photo (bruit aléatoire façon pellicule).
   Subtil, vit doucement comme une vraie image film, se dissout au survol. */
.card__media::before {
  content: ""; position: absolute; inset: -50%; z-index: 1; pointer-events: none;
  background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='160' height='160'%3E%3Cfilter id='g'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.9' numOctaves='2' stitchTiles='stitch'/%3E%3CfeColorMatrix type='saturate' values='0'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23g)'/%3E%3C/svg%3E");
  background-size: 160px 160px;
  mix-blend-mode: overlay;
  opacity: .09;
  transition: opacity .5s ease;
  animation: filmGrain .7s steps(4) infinite;
  will-change: transform;
}
@keyframes filmGrain {
  0%   { transform: translate(0, 0); }
  25%  { transform: translate(-3%, 2%); }
  50%  { transform: translate(2%, -3%); }
  75%  { transform: translate(-2%, -2%); }
  100% { transform: translate(3%, 3%); }
}
@media (prefers-reduced-motion: reduce) { .card__media::before { animation: none; } }
.card:hover .card__media::before, .card:focus-visible .card__media::before { opacity: .05; }
.card__play { position: absolute; z-index: 1; top: 50%; left: 50%; width: 56px; height: 56px; transform: translate(-50%,-50%) scale(.9); display: grid; place-items: center; border-radius: 50%; background: rgba(12,13,16,.55); backdrop-filter: blur(6px); border: 1px solid rgba(255,255,255,.25); color: #fff; font-size: 1.1rem; padding-left: 3px; opacity: 0; transition: .28s; }
.card:hover .card__play, .card:focus-visible .card__play { opacity: 1; transform: translate(-50%,-50%) scale(1); background: linear-gradient(120deg, var(--accent), var(--accent-2)); border-color: transparent; }
.card__body { padding: 15px 18px 18px; }
.card__body h3 { font-family: "Montserrat", sans-serif; margin: 0 0 4px; font-size: 1.18rem; font-weight: 700; }
.card__body p { margin: 0; color: var(--muted); font-size: .88rem; text-transform: uppercase; letter-spacing: .06em; }
.empty { grid-column: 1 / -1; text-align: center; color: var(--muted); padding: 40px 0; }

/* ============ LIGHTBOX ============ */
.lb { position: fixed; inset: 0; z-index: 5000; display: grid; place-items: center; padding: 24px; background: rgba(5,6,8,.5); backdrop-filter: blur(10px) saturate(1.1); -webkit-backdrop-filter: blur(10px) saturate(1.1); opacity: 0; transition: opacity .3s ease, background .3s ease; }
.lb[hidden] { display: none; }
.lb.is-open { opacity: 1; background: rgba(5,6,8,.82); }
.lb__frame { width: min(1040px, 96vw); transform: scale(.94) translateY(10px); opacity: 0; transition: transform .42s cubic-bezier(.16,.7,.18,1), opacity .32s ease; }
.lb.is-open .lb__frame { transform: none; opacity: 1; }
.lb__ratio { position: relative; width: 100%; aspect-ratio: 16 / 9; background: #000; border-radius: 16px 16px 0 0; overflow: hidden; box-shadow: 0 30px 90px rgba(0,0,0,.65); }
.lb__ratio iframe { position: absolute; inset: 0; width: 100%; height: 100%; border: 0; }
/* Bandeau infos sous la vidéo — liquid glass */
.lb__info { position: relative; overflow: hidden; background: rgba(20,22,28,.5); backdrop-filter: url(#lgDistort) blur(26px) saturate(1.6); -webkit-backdrop-filter: blur(26px) saturate(1.6); border: 1px solid rgba(255,255,255,.1); border-top: 0; border-radius: 0 0 18px 18px; padding: 20px 24px 22px; box-shadow: inset 0 1px 0 rgba(255,255,255,.14), 0 30px 90px rgba(0,0,0,.5); }
.lb__info::before { content: ""; position: absolute; left: 0; right: 0; top: 0; height: 2px; background: linear-gradient(90deg, var(--accent), var(--accent-2)); opacity: .85; }
.lb__info h2 { font-family: "Montserrat", sans-serif; margin: 0; font-size: clamp(1.3rem, 2.4vw, 1.75rem); font-weight: 800; letter-spacing: .01em; line-height: 1.1; }
.lb__meta { display: flex; align-items: center; gap: 10px; margin: 11px 0 0; flex-wrap: wrap; }
.lb__tag { font-size: .7rem; font-weight: 800; text-transform: uppercase; letter-spacing: .09em; color: #fff; padding: 5px 12px; border-radius: 999px; background: linear-gradient(120deg, var(--accent), var(--accent-2)); box-shadow: inset 0 1px 0 rgba(255,255,255,.3), 0 4px 14px rgba(108,123,255,.35); }
.lb__year { display: inline-flex; align-items: center; color: var(--muted); font-size: .85rem; font-weight: 600; letter-spacing: .04em; padding: 4px 12px; border-radius: 999px; background: rgba(255,255,255,.06); border: 1px solid rgba(255,255,255,.09); }
.lb__sub { margin: 12px 0 0; color: #c7ccd6; font-size: .97rem; line-height: 1.6; max-width: 70ch; }
.lb__close { position: absolute; top: 18px; right: 22px; width: 44px; height: 44px; border-radius: 50%; border: 1px solid rgba(255,255,255,.18); background: rgba(20,22,28,.45); backdrop-filter: url(#lgDistort) blur(16px) saturate(1.5); -webkit-backdrop-filter: blur(16px) saturate(1.5); box-shadow: inset 0 1px 0 rgba(255,255,255,.2); color: #fff; font-size: 1.1rem; cursor: pointer; transition: .25s; z-index: 3; }
.lb__close:hover { background: linear-gradient(120deg, var(--accent), var(--accent-2)); border-color: transparent; transform: rotate(90deg); }

@media (max-width: 560px) {
  .hero { padding-top: 34px; }
  .grid { grid-template-columns: 1fr; gap: 16px; }
}
