/* ============================================================================
   wld.gg — "amber phosphor / liminal ink"
   A cinematic text-adventure surface. Two worlds: a warm-amber CRT terminal
   (dark, the hero) and an old-novel paper mode (cream, Korean serif).
   ========================================================================== */

:root{
  --ease: cubic-bezier(.2,.7,.2,1);
  --ease-out: cubic-bezier(.16,1,.3,1);
  --mono: 'JetBrains Mono', ui-monospace, monospace;
  --kr-sans: 'IBM Plex Sans KR', 'Pretendard', sans-serif;
  --kr-serif: 'Gowun Batang', serif;
}
*{box-sizing:border-box;margin:0;padding:0;-webkit-tap-highlight-color:transparent}
/* the .screen/.scrim/.panel rules set display, which would override the UA
   [hidden] rule — make hidden authoritative so JS toggles actually hide. */
[hidden]{display:none!important}
html,body{height:100%}
/* stop iOS Safari from auto-inflating text in wide/non-wrapping blocks */
html{-webkit-text-size-adjust:100%;text-size-adjust:100%}
body{
  font-family:var(--kr-sans);
  overflow:hidden;position:relative;
  background:var(--bg);color:var(--fg);
  transition:background .8s var(--ease), color .8s var(--ease);
  -webkit-font-smoothing:antialiased;text-rendering:optimizeLegibility;
}
body.booting *{transition:none !important}
input,button,textarea{font-family:inherit;color:inherit}
textarea{resize:none;border:none;outline:none;background:transparent}
/* bound the app to the VISUAL viewport (set by JS via the VisualViewport API)
   so the bottom input dock stays above the on-screen keyboard instead of being
   hidden behind it. Falls back to 100dvh before JS runs. */
.screen{position:fixed;left:0;right:0;top:var(--app-top,0);height:var(--app-h,100dvh);
  display:flex;flex-direction:column;z-index:2}
/* themed surfaces morph smoothly when the World Seed picks the palette (the
   world "takes on its tone") instead of snapping dark↔light mid-experience */
#topbar,.story,#dock,.turn-row,.goal-pull,.statusbar,.chip,.title-chip,.icon-btn{
  transition:background-color .6s var(--ease),color .6s var(--ease),border-color .6s var(--ease)}
body.booting #topbar,body.booting .story,body.booting #dock{transition:none}

/* ---------- THEMES ---------- */
body.theme-terminal{
  --bg:#08070c; --bg-soft:#0f0d16; --panel:#100d18;
  --fg:#e9e2d4;             /* warm bone */
  --dim:#9a8f7a; --faint:#5f574a;
  --accent:#e8b066;         /* CRT amber */
  --accent-soft:#3a2c18;
  --npc:#79c8c0;            /* cool phosphor */
  --line:#1d1925; --line-2:#2a2434;
  --glow:rgba(232,176,102,.45);
  --story-font:var(--kr-sans); --story-spacing:.005em;
}
body.theme-paper{
  --bg:#efe7d6; --bg-soft:#f7f1e4; --panel:#f7f1e4;
  --fg:#2a241c;             /* ink */
  --dim:#867a64; --faint:#b3a386;
  --accent:#a8502b;         /* oxblood / burnt sienna */
  --accent-soft:#e8d6bd;
  --npc:#3f6f4f;            /* forest */
  --line:#e0d4bd; --line-2:#d3c4a8;
  --glow:rgba(168,80,43,.28);
  --story-font:var(--kr-serif); --story-spacing:.002em;
}

/* ---------- atmosphere layers ---------- */
#grain{position:fixed;inset:-50%;z-index:1;pointer-events:none;opacity:.045;
  background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='160' height='160'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='.85' numOctaves='2'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)'/%3E%3C/svg%3E");
  animation:grain 7s steps(6) infinite;mix-blend-mode:overlay}
@keyframes grain{0%{transform:translate(0,0)}20%{transform:translate(-3%,2%)}40%{transform:translate(2%,-3%)}60%{transform:translate(-2%,-2%)}80%{transform:translate(3%,3%)}100%{transform:translate(0,0)}}
#vignette{position:fixed;inset:0;z-index:1;pointer-events:none;
  background:radial-gradient(120% 90% at 50% 30%, transparent 55%, color-mix(in srgb, var(--bg) 88%, black) 100%)}
#scanlines{position:fixed;inset:0;z-index:1;pointer-events:none;opacity:0;
  background:repeating-linear-gradient(180deg, rgba(0,0,0,.16) 0 1px, transparent 1px 3px);
  transition:opacity .8s var(--ease)}
body.theme-terminal #scanlines{opacity:.35}
@media (prefers-reduced-motion:reduce){#grain{animation:none}}

/* ---------- COLD START ---------- */
#cold{justify-content:center;align-items:center}
.cold-wrap{width:100%;max-width:660px;padding:0 26px;margin-top:-7vh;position:relative}
/* pin the wordmark just under the browser chrome / notch with breathing room, so
   on mobile it's clearly visible at first instead of glued to (or clipped at) the
   very top — and it stays put regardless of the keyboard. It vanishes with the
   cold screen once a game starts. */
.cold-brand{position:fixed;top:calc(env(safe-area-inset-top) + 20px);left:24px;z-index:3;
  display:flex;align-items:center;gap:9px;
  font-family:var(--mono);font-size:13px;
  letter-spacing:.42em;text-transform:uppercase;color:var(--dim);
  animation:brandIn 1.6s var(--ease-out) .3s both}
.cold-brand .cold-mark{width:24px;height:24px;object-fit:contain;flex:0 0 auto;
  transition:filter .6s var(--ease)}
/* dark surface: the keycap glows on near-black as drawn. light (paper) surface:
   the amber keycap loses edge contrast on cream, so lift it with a soft shadow. */
body.theme-paper .cold-brand .cold-mark{
  filter:drop-shadow(0 2px 5px rgba(80,40,10,.42)) drop-shadow(0 0 1px rgba(80,40,10,.3))}
/* wordmark + tagline stacked — the tagline frames keyworld as a GAME (not a
   novel) at first glance, and vanishes with the cold screen once a game starts */
.cold-brand .cb-text{display:flex;flex-direction:column;gap:3px;min-width:0;line-height:1.12}
/* override the brand's wide .42em tracking so the wordmark reads "키월드" (attached),
   not "키 월 드" spaced apart */
.cold-brand .cb-name{letter-spacing:.02em}
.cold-brand .cb-tag{font-size:11px;letter-spacing:.02em;text-transform:none;
  color:var(--dim);white-space:nowrap}
@keyframes brandIn{from{opacity:0;letter-spacing:.7em}to{opacity:.92}}
.prompt-row{display:flex;align-items:center;width:100%}
.caret-line{display:flex;align-items:flex-start;width:100%;
  font-family:var(--story-font);font-size:clamp(22px,5.6vw,31px);line-height:1.5}
.prompt-mark{color:var(--accent);font-weight:400;margin-right:4px;transform:translateY(2px);
  animation:breathe 1.9s var(--ease) infinite;text-shadow:0 0 14px var(--glow)}
/* the ▏ prompt-mark IS the cursor (it breathes); hide the native caret so the
   cold screen shows a single, intentional blinking cursor per the design.
   It's a textarea so a long first line wraps to multiple lines instead of
   scrolling its front out of view. */
.cold-input{flex:1;background:transparent;border:none;outline:none;width:100%;
  font-family:var(--story-font);font-size:clamp(22px,5.6vw,31px);letter-spacing:-.01em;
  line-height:1.5;caret-color:transparent;margin-left:-2px;
  resize:none;overflow-y:auto;max-height:42vh;display:block}
.cold-input::placeholder{color:var(--faint)}
/* anchored to the bottom of the caret-line (top:100%), so its contents flow
   downward and never affect the vertically-centered caret above it */
.cold-below{position:absolute;top:100%;left:0;right:0;padding:0 26px}
/* margin-top + per-item padding kept slightly tighter than instinct so the block
   sits a touch higher — on iOS the autofill bar (key/card/location) rides up just
   above the keyboard and was crowding the lower examples. */
.suggestions{margin-top:28px;display:flex;flex-direction:column;gap:1px}
.suggestion{text-align:left;background:transparent;border:none;cursor:pointer;
  color:color-mix(in srgb,var(--fg) 64%,var(--dim));font-family:var(--story-font);
  font-size:clamp(16px,4.2vw,19px);padding:8px 0 8px 26px;position:relative;border-radius:8px;
  opacity:0;transform:translateY(8px);animation:fadeUp .7s var(--ease-out) forwards;
  transition:color .3s,padding-left .3s}
.suggestion::before{content:'┄';position:absolute;left:0;top:8px;color:var(--dim);
  transition:color .3s,transform .3s}
.suggestion:hover,.suggestion:active{color:var(--accent);padding-left:32px}
.suggestion:hover::before{color:var(--accent);transform:translateX(3px)}
@keyframes fadeUp{to{opacity:1;transform:none}}

/* first-turn world-building loader: a glowing keycap orb + cycling status lines,
   so the opening wait reads as a GAME world being authored (goals, characters,
   dangers) rather than a blank pause. */
.building{display:flex;align-items:center;gap:12px;padding:6px 0;margin-top:2px}
.bld-orb{flex:0 0 auto;width:18px;height:18px;border-radius:6px;
  background:radial-gradient(circle at 50% 38%,var(--accent),color-mix(in srgb,var(--accent) 24%,transparent));
  box-shadow:0 0 18px var(--glow);animation:bldOrb 1.5s var(--ease) infinite}
.bld-msg{font-family:var(--story-font);font-size:15.5px;color:var(--dim);
  transition:opacity .35s var(--ease)}
.bld-msg.bld-out{opacity:0}
@keyframes bldOrb{0%,100%{transform:scale(.82) rotate(0);opacity:.6}50%{transform:scale(1.08) rotate(8deg);opacity:1}}
@keyframes breathe{0%,100%{opacity:.25;transform:translateY(2px) scaleY(.92)}50%{opacity:1;transform:translateY(2px) scaleY(1)}}
@keyframes blink{50%{opacity:0}}

/* ---------- GAME ---------- */
#topbar{display:flex;align-items:center;justify-content:space-between;gap:8px;flex:0 0 auto;
  padding:max(12px,env(safe-area-inset-top)) 14px 10px;
  border-bottom:1px solid transparent;backdrop-filter:blur(2px)}
.title-chip{flex:1;text-align:center;font-family:var(--mono);font-size:13px;
  letter-spacing:.16em;text-transform:uppercase;color:var(--dim);
  white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:62%;margin:0 auto;opacity:.95;
  cursor:pointer;user-select:none;transition:color .2s,transform .15s}
.title-chip:hover{color:var(--fg)}
.title-chip:active{transform:scale(.97)}

/* floating "to newest" button — appears when you scroll up to re-read */
.to-bottom{position:absolute;left:50%;transform:translateX(-50%);bottom:calc(var(--dock-h,96px) + 14px);z-index:6;
  display:grid;place-items:center;width:38px;height:38px;border-radius:50%;
  background:var(--bg-2,var(--bg));color:var(--accent);border:1px solid var(--line-2);
  box-shadow:0 6px 20px rgba(0,0,0,.34);cursor:pointer;
  animation:rv .2s ease both;transition:transform .15s,border-color .2s,color .2s}
.to-bottom:hover{border-color:var(--accent);color:var(--fg)}
.to-bottom:active{transform:translateX(-50%) scale(.9)}
.to-bottom[hidden]{display:none}
.icon-btn{background:transparent;border:none;color:var(--dim);cursor:pointer;display:grid;place-items:center;
  width:40px;height:40px;border-radius:11px;transition:background .2s,color .2s,transform .2s;flex:0 0 auto}
.icon-btn:hover{background:var(--bg-soft);color:var(--accent)}
.icon-btn:active{transform:scale(.9)}

.story{flex:1;overflow-y:auto;overflow-x:hidden;padding:14px 22px 14px;
  font-family:var(--story-font);font-size:clamp(16.5px,4.5vw,18.5px);line-height:1.7;
  letter-spacing:var(--story-spacing);-webkit-overflow-scrolling:touch;scroll-behavior:smooth}
.story::-webkit-scrollbar{width:0}
.first-line{color:var(--dim);font-size:.92em;font-style:italic;opacity:.82;
  padding:4px 0 16px;margin-bottom:20px;position:relative}
.first-line::after{content:'';position:absolute;left:0;bottom:0;width:46px;height:1px;
  background:linear-gradient(90deg,var(--accent),transparent)}
.usr{color:var(--accent);margin:26px 0 8px;font-weight:600;font-family:var(--kr-sans);
  font-size:.94em;letter-spacing:.005em;display:flex;gap:8px;align-items:baseline;
  animation:slideRight .4s var(--ease-out)}
.usr::before{content:'›';opacity:.55;font-family:var(--mono)}
@keyframes slideRight{from{opacity:0;transform:translateX(-6px)}}
.blk{margin:0 0 .75em;white-space:pre-wrap;word-break:break-word}
.blk:last-child,.story-turn>:last-child{margin-bottom:0}
/* each new beat softly fades in → "스르르륵", not "다다다닥" */
.rv{animation:rv .55s var(--ease-out) both}
@keyframes rv{from{opacity:0;transform:translateY(2px)}to{opacity:1;transform:none}}
/* while streaming, softly fade the leading (newest) edge for a written-in feel */
.story-turn.streaming{-webkit-mask-image:linear-gradient(180deg,#000 calc(100% - 1.15em),transparent);
  mask-image:linear-gradient(180deg,#000 calc(100% - 1.15em),transparent)}
@media (prefers-reduced-motion:reduce){.rv{animation:none}.story-turn.streaming{-webkit-mask-image:none;mask-image:none}}
.npc-line{margin:13px 0;padding:2px 0 2px 15px;border-left:2px solid var(--npc);
  color:var(--fg);position:relative}
.npc-name{color:var(--npc);font-weight:700;margin-right:7px;font-family:var(--kr-sans);
  font-size:.92em;letter-spacing:.01em}
.story-turn:last-child{margin-bottom:8px}
.cursor-pulse::after{content:'▋';color:var(--accent);margin-left:1px;
  animation:blink 1s steps(1) infinite;text-shadow:0 0 10px var(--glow)}

.goal-pull{flex:0 0 auto;padding:3px 22px;color:var(--dim);font-size:13.5px;
  font-family:var(--kr-sans);letter-spacing:0;transition:opacity .5s;opacity:.95;line-height:1.7}
.goal-pull b{color:var(--accent);font-weight:600;opacity:.85}

/* responsive choice row (feedback 2026-06-01). DESKTOP: wrap — a mouse has no way
   to scroll a horizontal overflow, so we let chips flow onto ~2 rows; the wide
   screen has reading room to spare. MOBILE: single-row horizontal scroll (see the
   max-width:779px block) — touch swipes it natively, and keeping it to one row
   protects the story reading area from being eaten by stacked full-width chips. */
.suggest-actions{flex:0 0 auto;display:flex;flex-wrap:wrap;gap:7px;padding:9px 16px 3px}
.chip{flex:0 1 auto;max-width:100%;background:var(--bg-soft);border:1px solid var(--line-2);color:var(--dim);
  font-family:var(--kr-sans);font-size:13.5px;padding:8px 14px;border-radius:18px;cursor:pointer;
  white-space:normal;text-align:left;line-height:1.45;transition:transform .15s,color .2s,border-color .2s,background .2s;
  animation:chipIn .4s var(--ease-out) both}
.chip:nth-child(2){animation-delay:.05s}.chip:nth-child(3){animation-delay:.1s}
@keyframes chipIn{from{opacity:0;transform:translateY(6px)}}
.chip:active{transform:scale(.94)}
.chip:hover{color:var(--accent);border-color:var(--accent);background:var(--accent-soft)}

#dock{flex:0 0 auto;padding:4px 14px max(11px,env(safe-area-inset-bottom));
  background:linear-gradient(180deg,transparent,var(--bg) 22%);position:relative}
.statusbar{display:flex;gap:16px;align-items:center;flex-wrap:wrap;padding:6px 8px 9px;
  font-family:var(--mono);font-size:13px;color:var(--dim);cursor:pointer;min-height:15px;user-select:none}
.statusbar .stat{transition:color .35s,text-shadow .35s;white-space:nowrap}
.statusbar .stat.glow{color:var(--accent);text-shadow:0 0 14px var(--glow)}
.statusbar:empty::before{content:''}
.turn-row{background:var(--bg-soft);border:1px solid var(--line-2);border-radius:15px;
  padding:3px 5px 3px 15px;align-items:center;transition:border-color .3s,box-shadow .3s}
.turn-row:focus-within{border-color:var(--accent);box-shadow:0 0 0 3px var(--accent-soft)}
.turn-row .prompt-mark{color:var(--accent);font-family:var(--mono);font-size:17px;
  animation:none;margin-right:9px;text-shadow:0 0 10px var(--glow);transform:none}
.turn-input{flex:1;background:transparent;border:none;outline:none;
  font-family:var(--story-font);font-size:16.5px;line-height:1.45;padding:12px 0;caret-color:var(--accent);
  min-width:0;resize:none;overflow-y:auto;max-height:7.5em;display:block}
.turn-input::placeholder{color:var(--faint)}
.turn-input:disabled{opacity:.4}
.send-btn{background:var(--accent);color:var(--bg);border:none;width:42px;height:42px;
  border-radius:12px;cursor:pointer;flex:0 0 auto;display:grid;place-items:center;
  transition:transform .15s,opacity .2s,box-shadow .2s;box-shadow:0 4px 16px var(--glow)}
.send-btn:active{transform:scale(.9)}
.send-btn:disabled{opacity:.35;cursor:default;box-shadow:none}

/* ---------- PANEL ---------- */
.scrim{position:fixed;inset:0;background:color-mix(in srgb,var(--bg) 40%,rgba(0,0,0,.65));
  z-index:40;animation:fade .35s;backdrop-filter:blur(3px)}
@keyframes fade{from{opacity:0}}
.panel{position:fixed;top:0;right:0;bottom:0;width:min(360px,88vw);background:var(--panel);
  z-index:50;padding:26px 24px calc(26px + env(safe-area-inset-bottom));
  overflow-y:auto;overflow-x:hidden;overscroll-behavior:contain;-webkit-overflow-scrolling:touch;
  border-left:1px solid var(--line-2);box-shadow:-24px 0 70px rgba(0,0,0,.5);
  animation:slideIn .4s var(--ease-out)}
@keyframes slideIn{from{transform:translateX(100%)}}
.panel h3{font-family:var(--mono);font-size:11px;color:var(--accent);text-transform:uppercase;
  letter-spacing:.16em;margin:22px 0 10px}
.panel h3:first-child{margin-top:0}
.panel .row{display:flex;justify-content:space-between;gap:12px;padding:8px 0;font-size:14px;
  border-bottom:1px solid var(--line);line-height:1.5;align-items:flex-start}
/* status rows: short label fixed at left, long value wraps on the right */
.panel .row > span:first-child{flex:0 0 auto;white-space:nowrap}
.panel .row > span:last-child{flex:1;color:var(--dim);font-family:var(--mono);font-size:13px;
  text-align:right;word-break:break-word;min-width:0}
/* wrap rows (goals / inventory): long label wraps, short value fixed at right */
.panel .row.wrap > span:first-child{flex:1 1 auto;white-space:normal;word-break:break-word;min-width:0}
.panel .row.wrap > span:last-child{flex:0 0 auto;white-space:nowrap;text-align:right}
.panel .bar{height:5px;background:var(--line);border-radius:3px;overflow:hidden;margin:6px 0 2px}
.panel .bar>span{display:block;height:100%;background:linear-gradient(90deg,var(--accent),var(--npc));
  border-radius:3px;transition:width .6s var(--ease)}
.panel .empty{color:var(--faint);font-size:13px;font-style:italic}

/* ---------- GENERIC MODAL (library / confirm / share) ---------- */
/* modal sits ABOVE the ending card (z60) + its own scrim, so "다른 세계로 떠나기"
   and the share-link fallback layer on top of the ending instead of being buried
   behind it — closing the modal returns the player to the ending card, never a
   dead end (feedback 2026-06-01). */
#modal-scrim{z-index:65}
.modal{position:fixed;z-index:70;left:50%;top:50%;transform:translate(-50%,-50%);
  width:min(440px,92vw);max-height:86dvh;display:flex;flex-direction:column;
  /* scroll when content exceeds the cap (e.g. the account menu + 운영 콘솔 on a
     small phone) — was overflow:hidden, which clipped the bottom with no scroll */
  overflow-y:auto;overscroll-behavior:contain;-webkit-overflow-scrolling:touch;
  background:linear-gradient(165deg,var(--bg-soft),var(--bg));border:1px solid var(--line-2);
  border-radius:22px;box-shadow:0 36px 100px rgba(0,0,0,.6), inset 0 1px 0 color-mix(in srgb,var(--accent) 18%,transparent);
  padding:24px 22px;animation:popIn .42s var(--ease-out)}
.modal .m-head{font-family:var(--mono);font-size:12px;letter-spacing:.2em;text-transform:uppercase;
  color:var(--accent);margin-bottom:16px;display:flex;align-items:center;gap:8px}
.modal .m-head::before{content:'';width:20px;height:1px;background:var(--accent)}
.modal .m-body{font-size:15px;line-height:1.8;color:var(--dim);white-space:pre-wrap;margin-bottom:20px}
.modal .m-actions{display:flex;gap:10px}
.modal .m-btn{flex:1;padding:14px;border-radius:12px;font-weight:700;font-size:15px;cursor:pointer;
  border:none;font-family:var(--kr-sans);transition:transform .15s,opacity .2s}
.modal .m-btn:active{transform:scale(.95)}
.modal .m-btn.primary{background:var(--accent);color:var(--bg);box-shadow:0 6px 20px var(--glow)}
.modal .m-btn.ghost{background:transparent;color:var(--fg);border:1px solid var(--line-2)!important}

/* library */
.lib-new{width:100%;padding:15px;border:none;border-radius:14px;cursor:pointer;margin-bottom:16px;
  background:var(--accent);color:var(--bg);font-family:var(--kr-sans);font-weight:700;font-size:15.5px;
  box-shadow:0 6px 22px var(--glow);transition:transform .15s}
.lib-new:active{transform:scale(.97)}
.lib-list{display:flex;flex-direction:column;gap:9px;overflow-y:auto;flex:1;margin:0 -4px;padding:0 4px}
.lib-card{text-align:left;width:100%;background:var(--bg);border:1px solid var(--line);border-radius:14px;
  padding:14px 15px;cursor:pointer;transition:border-color .2s,transform .15s,background .2s;font-family:var(--kr-sans)}
.lib-card:hover{border-color:var(--accent);background:var(--bg-soft)}
.lib-card:active{transform:scale(.985)}
.lib-card.cur{border-color:var(--accent)}
.lib-top{display:flex;align-items:center;justify-content:space-between;gap:8px}
.lib-title{font-size:16px;font-weight:700;color:var(--fg);font-family:var(--kr-serif);
  white-space:nowrap;overflow:hidden;text-overflow:ellipsis}
.lib-now{flex:0 0 auto;font-family:var(--mono);font-size:10px;letter-spacing:.1em;color:var(--bg);
  background:var(--accent);padding:3px 8px;border-radius:999px}
.lib-meta{font-family:var(--mono);font-size:11.5px;color:var(--faint);margin-top:5px;letter-spacing:.02em}
.lib-snip{font-size:13px;color:var(--dim);margin-top:7px;line-height:1.55;
  display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}
.lib-empty{color:var(--faint);font-size:14px;font-style:italic;text-align:center;padding:24px 0}
.share-url{width:100%;background:var(--bg);border:1px solid var(--line-2);border-radius:12px;
  padding:13px;color:var(--fg);font-family:var(--mono);font-size:13px;margin-bottom:16px}

/* ---------- ENDING ---------- */
.ending-card{position:fixed;z-index:60;left:50%;top:50%;transform:translate(-50%,-50%);
  width:min(450px,93vw);max-height:90dvh;overflow-y:auto;
  background:linear-gradient(165deg,var(--bg-soft),var(--bg));
  border:1px solid var(--line-2);border-radius:24px;padding:34px 28px 26px;
  box-shadow:0 40px 110px rgba(0,0,0,.6), inset 0 1px 0 color-mix(in srgb,var(--accent) 25%,transparent);
  animation:popIn .6s var(--ease-out)}
@keyframes popIn{from{opacity:0;transform:translate(-50%,-46%) scale(.95)}}
.ending-card .kicker{font-family:var(--mono);font-size:10.5px;letter-spacing:.28em;color:var(--accent);
  text-transform:uppercase;margin-bottom:14px;display:flex;align-items:center;gap:8px}
.ending-card .kicker::before{content:'';width:22px;height:1px;background:var(--accent)}
.ending-card h1{font-family:var(--kr-serif);font-size:27px;line-height:1.32;margin-bottom:18px;
  letter-spacing:-.01em;font-weight:700}
.ending-card .summary{font-family:var(--story-font);font-size:15.5px;line-height:1.95;color:var(--dim);
  white-space:pre-wrap;margin-bottom:20px}
.ending-card .hl{font-size:13.5px;color:var(--accent);margin-bottom:18px;line-height:1.6}
.ending-card .result{display:flex;justify-content:space-between;align-items:center;gap:10px;
  padding:14px 16px;border-radius:14px;background:var(--bg);border:1px solid var(--line-2);margin-bottom:8px;
  font-weight:600}
.ending-card .rarity{font-family:var(--mono);font-size:12px;color:var(--dim)}
.ending-card .rarity b{color:var(--accent);font-size:14px}
.ending-card .stats{display:flex;gap:14px;flex-wrap:wrap;font-family:var(--mono);font-size:12px;
  color:var(--faint);padding:10px 2px 0;margin-bottom:18px}
.ending-card .actions{display:flex;gap:10px;margin-top:14px}
.ending-card .actions button{flex:1;padding:15px;border-radius:13px;font-weight:700;font-size:15px;
  cursor:pointer;border:none;transition:transform .15s,opacity .2s,box-shadow .2s;font-family:var(--kr-sans)}
.ending-card .actions button:active{transform:scale(.95)}
.btn-share{background:var(--accent);color:var(--bg);box-shadow:0 6px 22px var(--glow)}
.btn-restart{background:transparent;color:var(--fg);border:1px solid var(--line-2)!important}

/* "thinking" indicator: three dots that pulse in place — fixed width, no
   content-swapping, so it never jumps or blinks in/out */
.dots{display:flex;gap:6px;align-items:center;height:1.7em;margin:0}
.dots i{width:6px;height:6px;border-radius:50%;background:var(--accent);
  opacity:.22;animation:dotpulse 1.25s ease-in-out infinite;box-shadow:0 0 8px var(--glow)}
.dots i:nth-child(2){animation-delay:.16s}
.dots i:nth-child(3){animation-delay:.32s}
@keyframes dotpulse{0%,100%{opacity:.22;transform:translateY(0)}50%{opacity:1;transform:translateY(-3px)}}
@media (prefers-reduced-motion:reduce){.dots i{animation:none;opacity:.7}}

/* a quiet in-world note, not a push banner: translucent ink-glass with a thin
   amber hairline, so rewards/system lines whisper instead of shouting. */
.toast{position:fixed;bottom:calc(env(safe-area-inset-bottom) + 96px);left:50%;
  transform:translateX(-50%);
  background:color-mix(in srgb,var(--bg-soft) 86%,transparent);color:var(--fg);
  border:1px solid var(--accent-soft);padding:10px 16px;
  border-radius:13px;font-size:13px;font-weight:500;letter-spacing:.01em;line-height:1.45;z-index:80;
  width:max-content;max-width:calc(100vw - 20px);text-align:center;
  backdrop-filter:blur(8px);-webkit-backdrop-filter:blur(8px);
  animation:toastIn .5s var(--ease-out);box-shadow:0 8px 28px rgba(0,0,0,.35)}
@keyframes toastIn{from{opacity:0;transform:translate(-50%,14px)}}

/* desktop: a centered "reading column" framed like a page */
@media (min-width:780px){
  #topbar,.story,.goal-pull,#dock,.suggest-actions{max-width:760px;margin-left:auto;margin-right:auto;width:100%}
  #dock{border-left:1px solid var(--line);border-right:1px solid var(--line)}
  .story{padding-left:36px;padding-right:36px}
}

/* mobile: keep choices to ONE swipeable row so the story keeps its height.
   touch scrolls it natively; the edge fade hints there's more off-screen. */
@media (max-width:779px){
  .suggest-actions{flex-wrap:nowrap;overflow-x:auto;-webkit-overflow-scrolling:touch;
    scrollbar-width:none;mask-image:linear-gradient(90deg,transparent,#000 14px,#000 calc(100% - 14px),transparent)}
  .suggest-actions::-webkit-scrollbar{display:none}
  .chip{flex:0 0 auto;max-width:none;white-space:nowrap;border-radius:999px}
}

/* ===================== Phase 2 — accounts, credits, charge ===================== */

/* credit chip in the topbar */
.credits-chip{flex:0 0 auto;display:inline-flex;align-items:center;gap:4px;cursor:pointer;
  font-family:var(--mono);font-size:12.5px;font-weight:600;color:var(--dim);
  background:var(--bg-soft);border:1px solid var(--line-2);border-radius:999px;
  padding:5px 11px;line-height:1;transition:color .2s,border-color .2s,background .2s,box-shadow .3s}
.credits-chip b{color:var(--fg);font-weight:700}
.credits-chip:hover{color:var(--accent);border-color:var(--accent)}
.credits-chip:active{transform:scale(.94)}
.credits-chip.low{color:var(--accent);border-color:var(--accent);background:var(--accent-soft);
  box-shadow:0 0 0 3px color-mix(in srgb,var(--glow) 40%,transparent)}
.credits-chip.low b{color:var(--accent)}

/* modal copy bits */
.modal .m-lede{font-size:14.5px;line-height:1.7;color:var(--dim);margin:-4px 0 16px}
.modal .m-fine{font-size:12px;line-height:1.6;color:var(--faint);margin-top:14px}
.modal .m-free{list-style:none;margin:0 0 18px;padding:0;display:flex;flex-direction:column;gap:9px}
.modal .m-free li{font-size:13.5px;color:var(--dim);padding-left:20px;position:relative;line-height:1.5}
.modal .m-free li::before{content:'✦';position:absolute;left:0;color:var(--accent);opacity:.8}
.modal .m-free b{color:var(--fg)}
.modal .m-actions{display:flex;gap:10px;flex-wrap:wrap}
.modal .m-actions .m-btn{flex:1 1 auto}
.modal .m-row{display:flex;justify-content:space-between;align-items:center;
  padding:11px 0;border-bottom:1px solid var(--line);font-size:14px;color:var(--dim)}
.modal .m-row b{color:var(--fg);font-family:var(--mono)}
.modal .m-count{font-family:var(--mono);font-size:11px;color:var(--dim);margin-left:auto;letter-spacing:.05em}
/* stacked secondary buttons in the account sheet */
#modal .m-btn{display:block;width:100%;margin-top:10px;padding:13px;border-radius:12px;
  font-weight:700;font-size:14.5px;cursor:pointer;border:1px solid var(--line-2);
  background:var(--bg-soft);color:var(--fg);transition:transform .1s,border-color .2s}
#modal .m-btn:active{transform:scale(.97)}
#modal .m-btn.primary{background:var(--accent);color:var(--bg);border-color:var(--accent);box-shadow:0 6px 20px var(--glow)}
#modal .m-btn.ghost{background:transparent;color:var(--dim)}

/* Kakao login button (brand-correct yellow + brown) */
.kakao-btn{width:100%;display:flex;align-items:center;justify-content:center;gap:8px;
  background:#FEE500;color:#191600;border:none;border-radius:12px;padding:14px;
  font-family:var(--kr-sans);font-weight:700;font-size:15px;cursor:pointer;transition:transform .1s,filter .2s}
.kakao-btn:active{transform:scale(.97)}
.kakao-btn:hover{filter:brightness(1.03)}
.kakao-btn .k-ico{font-size:11px;transform:translateY(-1px)}

/* charge packs */
.pack-grid{display:grid;grid-template-columns:1fr 1fr;gap:10px;margin-top:4px}
.pack{position:relative;display:flex;flex-direction:column;gap:3px;align-items:flex-start;
  text-align:left;padding:14px 13px;border-radius:14px;cursor:pointer;
  background:var(--bg-soft);border:1px solid var(--line-2);transition:transform .1s,border-color .2s,box-shadow .25s}
.pack:active{transform:scale(.96)}
.pack:hover{border-color:var(--accent)}
.pack.hot{border-color:var(--accent);box-shadow:0 0 0 2px color-mix(in srgb,var(--glow) 35%,transparent)}
.pack-tag{position:absolute;top:-8px;right:10px;background:var(--accent);color:var(--bg);
  font-family:var(--mono);font-size:9.5px;font-weight:700;letter-spacing:.06em;padding:2px 7px;border-radius:999px}
.pack-name{font-family:var(--kr-sans);font-size:13px;color:var(--dim);font-weight:600}
.pack-cr{font-family:var(--mono);font-size:19px;font-weight:700;color:var(--fg)}
.pack-bonus{font-size:11.5px;color:var(--accent);font-weight:600;min-height:14px}
.pack-won{margin-top:5px;font-size:13px;color:var(--dim);font-weight:600}

/* achievements list */
.ach-list{display:flex;flex-direction:column;gap:2px;max-height:60vh;overflow-y:auto;margin:0 -4px}
.ach-row{display:flex;align-items:center;gap:11px;padding:11px 6px;border-bottom:1px solid var(--line);opacity:.55}
.ach-row.got{opacity:1}
.ach-mark{font-size:17px;flex:0 0 auto;width:22px;text-align:center}
.ach-body{flex:1 1 auto;display:flex;flex-direction:column;gap:2px;min-width:0}
.ach-body b{font-size:14px;color:var(--fg);font-weight:600}
.ach-body i{font-size:12px;color:var(--dim);font-style:normal;line-height:1.4}
.ach-rew{flex:0 0 auto;font-family:var(--mono);font-size:12px;color:var(--accent);font-weight:600}

/* welcome reward reveal */
.rw-list{display:flex;flex-direction:column;gap:2px;margin:4px 0 6px}
.rw-row{display:flex;justify-content:space-between;align-items:center;padding:12px 2px;
  border-bottom:1px solid var(--line);font-size:14.5px;color:var(--fg)}
.rw-row b{font-family:var(--mono);color:var(--accent);font-size:15px}
.rw-total{display:flex;justify-content:space-between;align-items:center;padding:13px 2px 4px;
  font-size:15px;font-weight:700;color:var(--fg)}
.rw-total b{font-family:var(--mono);color:var(--accent);font-size:19px;text-shadow:0 0 16px var(--glow)}
#modal .m-btn.primary#rw-go{margin-top:16px}

/* achievement reward — a real reward (credits), so it lands as a GIFT: a glowing
   card center-stage that pops in, holds, and fades. Tap to dismiss early. */
.reward{position:fixed;left:50%;top:38%;transform:translate(-50%,-50%);z-index:90;
  display:flex;flex-direction:column;align-items:center;gap:5px;cursor:pointer;
  padding:26px 36px 24px;border-radius:22px;text-align:center;max-width:calc(100vw - 40px);
  background:color-mix(in srgb,var(--bg-soft) 93%,transparent);
  border:1px solid var(--accent);
  box-shadow:0 20px 64px rgba(0,0,0,.5),0 0 0 1px var(--accent-soft) inset,0 0 44px var(--glow);
  backdrop-filter:blur(11px);-webkit-backdrop-filter:blur(11px);
  animation:rewardIn .6s var(--ease-out) both}
.reward.out{animation:rewardOut .45s var(--ease) forwards;pointer-events:none}
.reward-glow{position:absolute;inset:-32%;z-index:-1;pointer-events:none;
  background:radial-gradient(circle at 50% 36%,var(--glow),transparent 62%);
  animation:rewardPulse 2.4s var(--ease) infinite}
.reward-kicker{font-family:var(--mono);font-size:11px;letter-spacing:.34em;text-transform:uppercase;
  color:var(--accent)}
.reward-title{font-family:var(--story-font);font-size:23px;font-weight:600;color:var(--fg);letter-spacing:.01em}
.reward-amt{display:flex;align-items:baseline;gap:3px;margin-top:2px;
  color:var(--accent);text-shadow:0 0 20px var(--glow);
  /* delayed emphasis pop AFTER the card settles — draws the eye to the credits,
     reinforcing "I received something" (the whole point of this treatment) */
  animation:amtPop .55s var(--ease-out) .26s both}
.reward-plus{font-size:34px;font-weight:700;font-family:var(--mono)}
.reward-gem{font-size:17px}
/* KWCAG 5.3.1: muted --dim on the light (paper) card was 3.7:1 (< 4.5:1 for
   normal text) — lift toward --fg so it passes in both themes while staying
   clearly secondary to the title. */
.reward-desc{font-size:13px;color:color-mix(in srgb,var(--fg) 38%,var(--dim));
  max-width:240px;line-height:1.55;margin-top:3px}
@keyframes rewardIn{from{opacity:0;transform:translate(-50%,-46%) scale(.86)}}
@keyframes rewardOut{to{opacity:0;transform:translate(-50%,-54%) scale(.96)}}
@keyframes rewardPulse{0%,100%{opacity:.5;transform:scale(.95)}50%{opacity:.95;transform:scale(1.05)}}
@keyframes amtPop{0%{opacity:0;transform:translateY(5px) scale(.72)}60%{transform:translateY(0) scale(1.12)}100%{opacity:1;transform:none}}

/* in-game goal met — a light feel-good flourish (NO credits, just satisfaction):
   a popped checkmark + soft glow near the top. Lighter than the reward card. */
.goalcheer{position:fixed;top:calc(env(safe-area-inset-top) + 64px);left:50%;
  transform:translateX(-50%);z-index:85;display:flex;align-items:center;gap:11px;cursor:pointer;
  padding:11px 17px 11px 12px;border-radius:14px;max-width:calc(100vw - 28px);
  background:color-mix(in srgb,var(--bg-soft) 90%,transparent);
  border:1px solid var(--accent-soft);
  box-shadow:0 10px 34px rgba(0,0,0,.32),0 0 26px var(--glow);
  backdrop-filter:blur(9px);-webkit-backdrop-filter:blur(9px);
  animation:goalIn .55s var(--ease-out) both}
.goalcheer.out{animation:goalOut .4s var(--ease) forwards;pointer-events:none}
.gc-check{flex:0 0 auto;width:26px;height:26px;border-radius:50%;display:grid;place-items:center;
  font-size:14px;font-weight:700;color:var(--bg);background:var(--accent);
  box-shadow:0 0 16px var(--glow);animation:gcPop .5s var(--ease-out) .15s both}
.gc-body{display:flex;flex-direction:column;gap:1px;min-width:0}
.gc-kicker{font-family:var(--mono);font-size:11px;letter-spacing:.26em;text-transform:uppercase;color:var(--accent)}
.gc-desc{font-family:var(--story-font);font-size:14.5px;color:var(--fg);
  white-space:nowrap;overflow:hidden;text-overflow:ellipsis}
@keyframes goalIn{from{opacity:0;transform:translate(-50%,-14px)}}
@keyframes goalOut{to{opacity:0;transform:translate(-50%,-10px)}}
@keyframes gcPop{from{transform:scale(0);opacity:0}60%{transform:scale(1.25)}to{transform:scale(1)}}

/* objective cards — the opening "mission briefing" (and new goals mid-play).
   Distinct from .goalcheer (a completion): this announces what's at stake and
   what to DO, so the game reads as a game with a mission, not a novel. */
.objcard{position:fixed;top:calc(env(safe-area-inset-top) + 64px);left:50%;
  transform:translateX(-50%);z-index:86;display:flex;align-items:center;gap:12px;cursor:pointer;
  padding:12px 19px 12px 14px;border-radius:15px;max-width:calc(100vw - 28px);width:max-content;
  background:color-mix(in srgb,var(--bg-soft) 92%,transparent);
  border:1px solid var(--accent-soft);
  box-shadow:0 12px 38px rgba(0,0,0,.36),0 0 30px var(--glow);
  backdrop-filter:blur(10px);-webkit-backdrop-filter:blur(10px);
  animation:objIn .5s var(--ease-out) both}
.objcard.out{animation:objOut .4s var(--ease) forwards;pointer-events:none}
.objcard.stake{border-color:color-mix(in srgb,#e0604a 55%,var(--line-2))}
.obj-ic{flex:0 0 auto;font-size:20px;line-height:1;animation:objPop .55s var(--ease-out) .12s both;
  filter:drop-shadow(0 0 10px var(--glow))}
.obj-body{display:flex;flex-direction:column;gap:2px;min-width:0}
.obj-kicker{font-family:var(--mono);font-size:11px;letter-spacing:.3em;text-transform:uppercase;
  color:var(--accent);font-weight:600}
.objcard.stake .obj-kicker{color:#e0795f}
.obj-text{font-family:var(--story-font);font-size:15px;color:var(--fg);line-height:1.35;
  display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}
@keyframes objIn{from{opacity:0;transform:translate(-50%,-16px) scale(.96)}}
@keyframes objOut{to{opacity:0;transform:translate(-50%,-12px)}}
@keyframes objPop{from{transform:scale(0) rotate(-12deg);opacity:0}60%{transform:scale(1.25)}to{transform:scale(1)}}

/* ============================ Worlds · Leaderboard · Profile ============================
   Extends the "amber phosphor / liminal ink" language — a catalog of worlds and a
   ledger of legends, never generic game UI. */

.modal.wide{width:min(560px,94vw)}

/* segmented tabs — a manuscript index, not pill buttons */
.seg{display:flex;gap:4px;margin-bottom:18px;border-bottom:1px solid var(--line)}
.seg-b{flex:0 0 auto;background:none;border:none;border-bottom:2px solid transparent;
  cursor:pointer;font-family:var(--mono);font-size:12.5px;letter-spacing:.04em;font-weight:500;
  color:var(--dim);padding:8px 12px 9px;margin-bottom:-1px;transition:color .2s,border-color .2s}
.seg-b.on{color:var(--accent);font-weight:700;border-bottom-color:var(--accent)}
.seg-b:hover{color:var(--fg)}
.seg-b:active{transform:scale(.95)}

/* discovery feed — worlds as plates in a catalog */
.world-list{display:flex;flex-direction:column;gap:10px;overflow-y:auto;max-height:64dvh;margin:0 -2px;padding:2px}
.world-card{display:block;width:100%;text-align:left;cursor:pointer;
  background:var(--bg);border:1px solid var(--line-2);border-radius:14px;padding:15px 16px 13px;
  position:relative;transition:border-color .25s,transform .12s,box-shadow .3s}
.world-card::before{content:'';position:absolute;left:16px;right:16px;top:0;height:1px;
  background:linear-gradient(90deg,var(--accent),transparent);opacity:.5}
.world-card:hover{border-color:var(--accent);box-shadow:0 8px 26px rgba(0,0,0,.3)}
.world-card:active{transform:scale(.99)}
.wc-genre{font-family:var(--mono);font-size:10.5px;letter-spacing:.16em;text-transform:uppercase;color:var(--accent);opacity:.85;margin-bottom:6px}
.wc-title{font-family:var(--kr-serif);font-size:18px;font-weight:700;color:var(--fg);line-height:1.3;margin-bottom:6px}
.wc-premise{font-family:var(--kr-serif);font-size:13.5px;line-height:1.6;color:var(--dim);
  display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden;margin-bottom:10px}
.wc-foot{display:flex;justify-content:space-between;font-family:var(--mono);font-size:11px;color:var(--faint)}

/* world preview — standing at the threshold */
.wp-title{font-family:var(--kr-serif);font-size:24px;font-weight:700;color:var(--fg);line-height:1.3;margin:2px 0 8px}
.wp-meta{font-family:var(--mono);font-size:11.5px;color:var(--faint);margin-bottom:16px}
.wp-opening{font-family:var(--kr-serif);font-size:15px;line-height:1.85;color:var(--dim);
  white-space:pre-wrap;max-height:42dvh;overflow-y:auto;margin-bottom:18px;
  padding-left:14px;border-left:2px solid var(--accent-soft)}

/* leaderboard — an engraved register */
.lb-seg{flex-wrap:wrap}
.lb-list{display:flex;flex-direction:column;overflow-y:auto;max-height:62dvh;margin:0 -2px}
.lb-row{display:flex;align-items:center;gap:12px;padding:11px 6px;border-bottom:1px solid var(--line)}
.lb-rank{flex:0 0 32px;text-align:center;font-family:var(--mono);font-size:14px;font-weight:700;color:var(--dim)}
.lb-row:nth-child(1) .lb-rank,.lb-row:nth-child(2) .lb-rank,.lb-row:nth-child(3) .lb-rank{color:var(--accent);text-shadow:0 0 12px var(--glow)}
.lb-name{flex:1 1 auto;display:flex;flex-direction:column;gap:1px;min-width:0}
.lb-name b{font-size:14.5px;color:var(--fg);font-weight:600;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}
.lb-name i{font-family:var(--mono);font-size:10.5px;font-style:normal;letter-spacing:.08em;color:var(--accent);opacity:.8}
.lb-val{flex:0 0 auto;font-family:var(--mono);font-size:13.5px;font-weight:600;color:var(--fg)}
.lb-row.me{background:var(--accent-soft);border-radius:10px;border-bottom-color:transparent}
.lb-row.me .lb-name b{color:var(--accent)}
.lb-me{margin-top:8px;padding-top:8px;border-top:1px dashed var(--line-2)}

/* brag profile — a seal / crest */
.pf-crest{text-align:center;padding:8px 0 18px;border-bottom:1px solid var(--line);margin-bottom:18px}
.pf-title{display:inline-block;background:var(--accent-soft);color:var(--accent);
  font-family:var(--mono);font-size:12px;font-weight:700;letter-spacing:.12em;
  padding:6px 15px;border-radius:999px;margin-bottom:12px}
.pf-name{font-family:var(--kr-serif);font-size:24px;font-weight:700;color:var(--fg);margin-bottom:6px}
.pf-rank{font-size:13.5px;color:var(--dim)}.pf-rank b{color:var(--accent);font-family:var(--mono);font-size:16px}
.pf-stats{display:grid;grid-template-columns:repeat(3,1fr);gap:8px;margin-bottom:18px}
.pf-stat{background:var(--bg);border:1px solid var(--line-2);border-radius:12px;padding:11px 8px;text-align:center}
.pf-stat span{display:block;font-size:10.5px;color:var(--faint);margin-bottom:4px}
.pf-stat b{font-family:var(--mono);font-size:17px;color:var(--accent)}
.pf-sec{font-family:var(--mono);font-size:11px;letter-spacing:.14em;text-transform:uppercase;color:var(--dim);margin:6px 0 10px}
.pf-achs{display:flex;flex-wrap:wrap;gap:6px;margin-bottom:18px}
.pf-ach{font-size:12px;color:var(--fg);background:var(--bg);border:1px solid var(--accent-soft);border-radius:8px;padding:5px 10px}
.pf-worlds{display:flex;flex-direction:column;gap:6px;margin-bottom:18px}
.pf-world{display:flex;justify-content:space-between;align-items:center;cursor:pointer;
  background:var(--bg);border:1px solid var(--line-2);border-radius:10px;padding:11px 13px;text-align:left;transition:border-color .2s}
.pf-world:hover{border-color:var(--accent)}
.pf-world b{font-family:var(--kr-serif);font-size:14.5px;color:var(--fg);font-weight:600}
.pf-world span{font-family:var(--mono);font-size:11px;color:var(--faint)}

/* library + ending-card discovery entries */
.lib-discover,.ec-link{width:100%;background:none;border:none;cursor:pointer;
  font-family:var(--mono);font-size:12.5px;color:var(--accent);opacity:.85;padding:10px;
  letter-spacing:.04em;transition:opacity .2s}
.lib-discover{border:1px dashed var(--line-2);border-radius:12px;margin-bottom:16px}
.lib-discover:hover,.ec-link:hover{opacity:1}
.ec-more{display:flex;flex-direction:column;gap:2px;margin-top:14px;padding-top:12px;border-top:1px solid var(--line)}

/* two-path charge nudge */
.fork-paths{display:flex;flex-direction:column;gap:8px;margin:4px 0 16px}
.fp{background:var(--bg);border:1px solid var(--line-2);border-radius:12px;padding:12px 14px}
.fp b{display:block;font-size:14.5px;color:var(--fg);margin-bottom:3px}
.fp span{font-size:12.5px;color:var(--dim);line-height:1.5}

/* free story reader — read someone's finished story in their world's tone */
.rd-body{max-height:56dvh;overflow-y:auto;margin:2px -2px 16px;padding:0 4px;
  font-family:var(--story-font);font-size:15px;line-height:1.8}
.rd-usr{color:var(--accent);font-weight:600;margin:18px 0 7px;font-family:var(--kr-sans);font-size:13.5px}
.rd-usr::before{content:'› ';opacity:.65}
.rd-story{color:var(--fg);margin-bottom:11px}
/* give the reader a little breathing room at the bottom so the last line never
   sits flush against the fixed CTA. */
.rd-body{padding-bottom:6px}

/* ---------- refined scrollbars ----------
   The default UA scrollbar (a chunky pale bar) broke the cinematic surface —
   most visibly on the story-reader modal. Everywhere a modal area scrolls, we
   swap it for a thin, inset, rounded thumb tinted to the LIVE theme's accent
   (amber on terminal, oxblood on paper): nearly invisible at rest, brightening
   on hover. Firefox via scrollbar-width/-color; Chromium/Safari via ::-webkit. */
.modal,.rd-body,.world-list,.lb-list,.lib-list,.ach-list,.menu,.wp-opening{
  scrollbar-width:thin;
  scrollbar-color:color-mix(in srgb,var(--accent) 34%,var(--line)) transparent}
.modal::-webkit-scrollbar,.rd-body::-webkit-scrollbar,.world-list::-webkit-scrollbar,
.lb-list::-webkit-scrollbar,.lib-list::-webkit-scrollbar,.ach-list::-webkit-scrollbar,
.menu::-webkit-scrollbar,.wp-opening::-webkit-scrollbar{width:10px;height:10px}
.modal::-webkit-scrollbar-track,.rd-body::-webkit-scrollbar-track,.world-list::-webkit-scrollbar-track,
.lb-list::-webkit-scrollbar-track,.lib-list::-webkit-scrollbar-track,.ach-list::-webkit-scrollbar-track,
.menu::-webkit-scrollbar-track,.wp-opening::-webkit-scrollbar-track{background:transparent}
/* the 3px transparent border + padding-box clip floats a ~4px thumb inside the
   10px track, so it reads as a slim pill with breathing room, not a fat bar. */
.modal::-webkit-scrollbar-thumb,.rd-body::-webkit-scrollbar-thumb,.world-list::-webkit-scrollbar-thumb,
.lb-list::-webkit-scrollbar-thumb,.lib-list::-webkit-scrollbar-thumb,.ach-list::-webkit-scrollbar-thumb,
.menu::-webkit-scrollbar-thumb,.wp-opening::-webkit-scrollbar-thumb{
  background:color-mix(in srgb,var(--accent) 32%,var(--line));
  border:3px solid transparent;border-radius:99px;background-clip:padding-box}
.modal::-webkit-scrollbar-thumb:hover,.rd-body::-webkit-scrollbar-thumb:hover,.world-list::-webkit-scrollbar-thumb:hover,
.lb-list::-webkit-scrollbar-thumb:hover,.lib-list::-webkit-scrollbar-thumb:hover,.ach-list::-webkit-scrollbar-thumb:hover,
.menu::-webkit-scrollbar-thumb:hover,.wp-opening::-webkit-scrollbar-thumb:hover{
  background:color-mix(in srgb,var(--accent) 60%,var(--line));background-clip:padding-box}

/* operator console entry (the console itself is the dedicated /admin page) */
.admin-entry{color:var(--accent)!important}
