:root{--sans: "Inter", ui-sans-serif, system-ui, -apple-system, sans-serif;--mono: ui-monospace, "SF Mono", Menlo, monospace;--r-sm: 4px;--r-md: 8px;--r-lg: 14px;--t-fast: .14s cubic-bezier(.2,.6,.2,1);--t-med: .24s cubic-bezier(.2,.6,.2,1);--hairline: .5px}[data-theme=warm-light]{--bg: oklch(96% .022 130);--bg-2: oklch(93% .026 130);--bg-3: oklch(89% .029 130);--ink: oklch(22% .02 140);--ink-2: oklch(40% .02 140);--ink-3: oklch(56% .022 140);--rule: oklch(85% .029 130);--accent: oklch(50% .14 145);--accent-soft: oklch(50% .14 145 / .14);--good: oklch(58% .14 145)}[data-theme=warm-dark]{--bg: oklch(10% .019 145);--bg-2: oklch(13% .022 145);--bg-3: oklch(18% .024 145);--ink: oklch(94% .014 130);--ink-2: oklch(72% .018 130);--ink-3: oklch(50% .02 135);--rule: oklch(22% .026 145);--accent: oklch(72% .18 140);--accent-soft: oklch(72% .18 140 / .18);--good: oklch(74% .14 145)}@media print{.sidebar,.sysline{display:none!important}.app{grid-template-columns:1fr!important}.page{padding:0!important;max-width:100%!important}body{background:#fff!important;color:#000!important}}*,*:before,*:after{box-sizing:border-box}html,body{margin:0;padding:0}body{font-family:var(--sans);font-size:15px;line-height:1.55;-webkit-text-size-adjust:100%;color:var(--ink);background:var(--bg);-webkit-font-smoothing:antialiased;text-rendering:optimizeLegibility;font-feature-settings:"ss01","cv11";transition:background var(--t-med),color var(--t-med)}a{color:inherit;text-decoration:none}button{font:inherit;color:inherit;background:none;border:0;padding:0;cursor:pointer}input,textarea{font:inherit;color:inherit}::selection{background:var(--accent-soft);color:var(--ink)}.app{min-height:100vh;display:grid;grid-template-columns:260px minmax(0,1fr)}@media(max-width:880px){.app{grid-template-columns:1fr}}.sidebar{border-right:var(--hairline) solid var(--rule);padding:24px 20px;display:flex;flex-direction:column;gap:28px;position:sticky;top:0;height:100vh;overflow-y:auto}@media(max-width:880px){.sidebar{position:static;height:auto;border-right:0;border-bottom:var(--hairline) solid var(--rule);padding:16px 20px}}.brand{display:flex;flex-direction:column;gap:4px}.brand-name{font-family:var(--sans);font-weight:600;font-size:18px;letter-spacing:-.01em;line-height:1.1}.brand-role{color:var(--ink-2);font-size:13px;margin-top:2px}.nav{display:flex;flex-direction:column;gap:2px}.nav-item{display:grid;grid-template-columns:14px 1fr auto;align-items:center;gap:10px;padding:9px 10px;border-radius:var(--r-md);color:var(--ink-2);cursor:pointer;transition:background var(--t-fast),color var(--t-fast);font-size:13.5px;min-height:36px}.nav-item .glyph{font-family:var(--mono);font-size:10px;color:var(--ink-3);text-align:center}.nav-item .meta{font-family:var(--mono);font-size:10px;color:var(--ink-3)}.nav-item:hover{background:var(--bg-2);color:var(--ink)}.nav-item.is-active{background:var(--bg-3);color:var(--ink)}.nav-item.is-active .glyph{color:var(--accent)}.sidebar-foot{margin-top:auto;font-size:12.5px;color:var(--ink-3)}.foot-icons{display:flex;align-items:center;gap:6px;padding-top:12px;border-top:var(--hairline) solid var(--rule)}.icon-btn{display:inline-flex;align-items:center;justify-content:center;width:32px;height:32px;border-radius:var(--r-md);color:var(--ink-3);background:transparent;transition:background var(--t-fast),color var(--t-fast),box-shadow var(--t-fast)}.icon-btn svg{width:16px;height:16px;display:block}.icon-btn:hover{color:var(--ink);background:var(--bg-2)}.icon-btn[aria-pressed=true]{color:var(--ink);background:var(--bg-2);box-shadow:0 0 0 .5px var(--rule)}.main{min-width:0}.page{padding:64px 80px 96px;max-width:1180px}@media(max-width:1100px){.page{padding:56px 56px 80px;max-width:100%}}@media(max-width:880px){.page{padding:32px 24px 64px}}.kicker{font-family:var(--mono);font-size:10.5px;letter-spacing:.14em;text-transform:uppercase;color:var(--ink-3);display:flex;align-items:center;gap:10px}.kicker:before{content:"";width:18px;height:1px;background:var(--accent);display:inline-block}h1.page-title{font-family:var(--sans);font-weight:600;font-size:clamp(30px,4.6vw,48px);line-height:1.08;letter-spacing:-.022em;margin:14px 0 22px;text-wrap:balance}h1.page-title em{font-style:normal;color:var(--ink-2);font-weight:400}.lede{font-size:18px;line-height:1.55;color:var(--ink-2);max-width:64ch;text-wrap:pretty}.divider{border:0;border-top:var(--hairline) solid var(--rule);margin:48px 0}.btn{display:inline-flex;align-items:center;gap:8px;padding:9px 14px;border:var(--hairline) solid var(--rule);border-radius:999px;font-size:13px;background:var(--bg);transition:background var(--t-fast),border-color var(--t-fast),transform var(--t-fast)}.btn:hover{background:var(--bg-2);border-color:var(--ink-3)}.btn:active{transform:translateY(1px)}.btn--primary{background:var(--ink);color:var(--bg);border-color:var(--ink)}.btn--primary:hover{background:var(--ink-2);border-color:var(--ink-2)}.btn .icon{font-family:var(--mono);font-size:11px;opacity:.7}.chip{display:inline-flex;align-items:center;gap:4px;padding:3px 9px;border-radius:999px;background:var(--bg-2);border:var(--hairline) solid var(--rule);font-family:var(--mono);font-size:10.5px;color:var(--ink-2);letter-spacing:.02em;cursor:pointer}.chip[data-active=true]{background:var(--accent-soft);border-color:color-mix(in oklab,var(--accent) 40%,transparent);color:var(--ink)}.section-head{display:flex;align-items:baseline;justify-content:space-between;margin:56px 0 16px;gap:12px}.section-head h2{font-family:var(--sans);font-weight:600;font-size:20px;letter-spacing:-.012em;margin:0}.section-head .seg{font-family:var(--mono);font-size:10.5px;color:var(--ink-3);letter-spacing:.08em;text-transform:uppercase}.index-table{font-family:var(--mono);font-size:13px;border-top:var(--hairline) solid var(--rule)}.index-row{display:grid;grid-template-columns:28px 1fr 110px 80px;gap:16px;padding:14px 8px;border-bottom:var(--hairline) solid var(--rule);align-items:center;cursor:pointer;transition:background var(--t-fast);color:inherit}.index-row:hover{background:var(--bg-2)}.index-row .ix{color:var(--ink-3)}.index-row .name{color:var(--ink);font-weight:500}.index-row .name small{font-family:var(--sans);color:var(--ink-3);font-weight:400;margin-left:8px}.index-row .meta{color:var(--ink-3)}.index-row .arrow{color:var(--ink-3);text-align:right;transition:color var(--t-fast),transform var(--t-fast)}.index-row:hover .arrow{color:var(--accent);transform:translate(4px)}.index-row.is-disabled{cursor:default;opacity:.7}.index-row.is-disabled:hover{background:transparent}.index-row.is-disabled:hover .arrow{color:var(--ink-3);transform:none}@media(max-width:720px){.index-row{grid-template-columns:24px 1fr 60px}.index-row .meta{display:none}}.proj-layout{display:grid;grid-template-columns:220px minmax(0,1fr);gap:56px;margin-top:12px}@media(max-width:880px){.proj-layout{grid-template-columns:1fr;gap:20px}}.tree{font-family:var(--mono);font-size:12px;color:var(--ink-2);align-self:start;position:sticky;top:24px}@media(max-width:880px){.tree{position:static}}.tree-head{font-size:10.5px;letter-spacing:.14em;text-transform:uppercase;color:var(--ink-3);margin-bottom:8px}.tree-group{margin-bottom:12px}.tree-group-label{display:flex;align-items:center;gap:6px;color:var(--ink-3);cursor:pointer;padding:4px 0;user-select:none}.tree-group-label .caret{display:inline-block;width:10px;transition:transform var(--t-fast)}.tree-group[data-open=false] .caret{transform:rotate(-90deg)}.tree-list{list-style:none;padding:0 0 0 14px;margin:2px 0 0;border-left:var(--hairline) solid var(--rule);display:flex;flex-direction:column;gap:1px;overflow:hidden}.tree-group[data-open=false] .tree-list{display:none}.tree-leaf{display:flex;align-items:center;gap:8px;padding:6px 8px 6px 10px;border-radius:var(--r-sm);color:var(--ink-2);cursor:pointer;position:relative;transition:background var(--t-fast),color var(--t-fast);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;min-height:32px}.tree-leaf:before{content:"─";color:var(--ink-3);margin-right:-2px}.tree-leaf:hover{background:var(--bg-2);color:var(--ink)}.tree-leaf.is-active{background:var(--accent-soft);color:var(--ink)}.tree-leaf .lang{font-size:10px;color:var(--ink-3);margin-left:auto}.proj-body{min-width:0}.proj-header{display:flex;flex-direction:column;gap:10px;margin-bottom:18px}.proj-id{display:flex;gap:10px;align-items:center;flex-wrap:wrap;font-family:var(--mono);font-size:11px;color:var(--ink-3);letter-spacing:.02em}.proj-id .badge{padding:2px 6px;border:var(--hairline) solid var(--rule);border-radius:var(--r-sm)}.proj-id .status{color:var(--accent)}.proj-title{font-family:var(--sans);font-size:30px;font-weight:600;letter-spacing:-.018em;line-height:1.15;margin:0}.proj-tags{display:flex;gap:6px;flex-wrap:wrap;margin-top:6px}.proj-media{border:var(--hairline) solid var(--rule);border-radius:var(--r-lg);background:var(--bg-2);aspect-ratio:21 / 9;display:grid;place-items:center;position:relative;overflow:hidden;margin:18px 0 22px}.proj-media .placeholder{font-family:var(--mono);font-size:11px;color:var(--ink-3);letter-spacing:.08em;text-transform:uppercase;z-index:1}.proj-media:before{content:"";position:absolute;inset:0;background:repeating-linear-gradient(135deg,transparent 0 14px,color-mix(in oklab,var(--ink) 4%,transparent) 14px 15px);opacity:.8}.proj-prose{max-width:68ch;color:var(--ink-2);font-size:15px}.proj-prose p{margin:0 0 14px;text-wrap:pretty}.proj-prose p strong{color:var(--ink);font-weight:500}.proj-meta-grid{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:16px 32px;margin:24px 0;padding:18px;border:var(--hairline) solid var(--rule);border-radius:var(--r-lg);background:var(--bg-2)}@media(max-width:600px){.proj-meta-grid{grid-template-columns:1fr}}.proj-meta-grid .k{font-family:var(--mono);font-size:10px;letter-spacing:.12em;text-transform:uppercase;color:var(--ink-3);margin-bottom:4px}.proj-meta-grid .v{font-size:13.5px}.proj-actions{display:flex;gap:8px;flex-wrap:wrap;margin-top:12px}.resume-grid{display:grid;grid-template-columns:180px minmax(0,1fr);gap:32px;margin-top:8px}@media(max-width:720px){.resume-grid{grid-template-columns:1fr;gap:8px}}.resume-label{font-family:var(--mono);font-size:10.5px;letter-spacing:.14em;text-transform:uppercase;color:var(--ink-3);padding-top:6px}.resume-body{display:flex;flex-direction:column;gap:22px}.resume-row{border-top:var(--hairline) solid var(--rule);padding-top:16px;display:flex;flex-direction:column;gap:4px}.resume-row:first-child{border-top:0;padding-top:0}.resume-row .role{font-family:var(--sans);font-size:17px;font-weight:600;letter-spacing:-.005em}.resume-row .at{font-family:var(--sans);color:var(--ink-2);font-size:14px;display:flex;justify-content:space-between;flex-wrap:wrap;gap:8px}.resume-row .at .when{font-family:var(--mono);font-size:11px;color:var(--ink-3)}.resume-row ul{list-style:none;margin:6px 0 0;padding:0;display:flex;flex-direction:column;gap:4px}.resume-row li{color:var(--ink-2);font-size:14px;padding-left:14px;position:relative;text-wrap:pretty}.resume-row li:before{content:"▸";position:absolute;left:0;top:0;color:var(--accent);font-family:var(--mono);font-size:10px;line-height:1.7}.resume-headline{font-family:var(--mono);font-size:12px;letter-spacing:.04em;color:var(--ink-2);margin:-8px 0 18px;max-width:64ch;text-wrap:pretty}.skill-cats{display:flex;flex-direction:column;gap:0}.skill-cat{display:grid;grid-template-columns:180px 1fr;gap:16px;align-items:baseline;border-bottom:var(--hairline) solid var(--rule);padding:10px 0}.skill-cat:last-child{border-bottom:0}.skill-cat-name{font-family:var(--mono);font-size:11px;color:var(--ink-3);text-transform:uppercase;letter-spacing:.08em}.skill-cat-items{color:var(--ink);font-size:13.5px;line-height:1.6}@media(max-width:600px){.skill-cat{grid-template-columns:1fr;gap:4px;padding:8px 0}}.hero-actions{display:flex;gap:8px;flex-wrap:wrap;margin-top:6px}.blog-controls{display:flex;gap:8px;flex-wrap:wrap;align-items:center;margin:8px 0 18px}.search{flex:1 1 200px;display:flex;align-items:center;gap:8px;border:var(--hairline) solid var(--rule);border-radius:999px;padding:8px 14px;background:var(--bg);transition:border-color var(--t-fast);min-width:0}.search:focus-within{border-color:var(--ink-3)}.search .icon{font-family:var(--mono);font-size:12px;color:var(--ink-3)}.search input{flex:1;border:0;background:transparent;outline:0;font-size:14px;min-width:0}.tag-row{display:flex;gap:6px;flex-wrap:wrap}.tag-row .chip{transition:background var(--t-fast),border-color var(--t-fast)}.tag-row .chip:hover{border-color:var(--ink-3)}.posts{border-top:var(--hairline) solid var(--rule)}.post-year{font-family:var(--mono);font-size:11px;color:var(--ink-3);letter-spacing:.12em;padding:18px 0 8px}.post-row{display:grid;grid-template-columns:96px 1fr auto;gap:18px;align-items:baseline;padding:16px 8px 16px 0;border-bottom:var(--hairline) solid var(--rule);cursor:pointer;transition:background var(--t-fast),padding var(--t-fast);color:inherit}.post-row:hover{background:var(--bg-2);padding-left:8px}.post-row .when{font-family:var(--mono);font-size:11px;color:var(--ink-3)}.post-row .title{font-family:var(--sans);font-size:17px;font-weight:600;letter-spacing:-.005em;line-height:1.3}.post-row .title small{display:block;font-family:var(--sans);font-size:13.5px;color:var(--ink-2);margin-top:4px;font-weight:400;text-wrap:pretty}.post-row .read{font-family:var(--mono);font-size:10.5px;color:var(--ink-3);white-space:nowrap}.post-row .post-tags{grid-column:2 / 3;margin-top:6px;display:flex;gap:4px;flex-wrap:wrap}.posts-empty{padding:32px 0;color:var(--ink-3);font-family:var(--mono);font-size:12px}@media(max-width:600px){.post-row{grid-template-columns:1fr auto}.post-row .when,.post-row .post-tags{grid-column:1 / -1}}.sysline{margin-top:80px;padding-top:16px;border-top:var(--hairline) solid var(--rule);display:flex;justify-content:space-between;flex-wrap:wrap;gap:8px;font-family:var(--mono);font-size:10.5px;color:var(--ink-3);letter-spacing:.04em}.sysline .blink{color:var(--accent);animation:blink 1.6s steps(2,end) infinite}@keyframes blink{0%,49%{opacity:1}50%,to{opacity:0}}.post-detail{max-width:760px}.post-detail-back{font-family:var(--mono);font-size:11px;color:var(--ink-3);letter-spacing:.04em;margin-bottom:8px;display:inline-block;transition:color var(--t-fast)}.post-detail-back:hover{color:var(--accent)}.post-body{color:var(--ink-2);font-size:16px;line-height:1.7;max-width:68ch}.post-body p{margin:0 0 18px;text-wrap:pretty}.post-body h2{font-family:var(--sans);font-size:18px;font-weight:600;color:var(--ink);margin:32px 0 12px;letter-spacing:-.01em}.post-body h3{font-family:var(--sans);font-size:16px;font-weight:600;color:var(--ink);margin:28px 0 10px;letter-spacing:-.005em}.post-body code{font-family:var(--mono);font-size:13.5px;background:var(--bg-2);border:var(--hairline) solid var(--rule);border-radius:var(--r-sm);padding:1px 5px}.post-body pre{font-family:var(--mono);font-size:13px;background:var(--bg-2);border:var(--hairline) solid var(--rule);border-radius:var(--r-md);padding:14px 16px;overflow-x:auto;margin:0 0 18px}.post-body pre code{background:transparent;border:0;padding:0}.post-body ul,.post-body ol{margin:0 0 18px;padding-left:22px}.post-body blockquote{margin:0 0 18px;padding:4px 16px;border-left:2px solid var(--accent);color:var(--ink-3);font-style:italic}.post-foot{margin-top:40px;padding-top:24px;border-top:var(--hairline) solid var(--rule)}@media(max-width:880px){.sidebar{display:grid;grid-template-columns:auto 1fr;gap:12px 16px;padding:14px 16px;align-items:center}.sidebar .brand{grid-column:1}.sidebar .nav{grid-column:2;flex-direction:row;flex-wrap:wrap;justify-content:flex-end;gap:4px}.sidebar .nav-item{grid-template-columns:auto auto;padding:8px 12px;font-size:13px}.sidebar .nav-item .meta,.sidebar-foot,.brand-role{display:none}}@media(max-width:560px){.sidebar{grid-template-columns:1fr}.sidebar .nav{justify-content:flex-start}}@media(max-width:720px){.page{padding:28px 20px 56px}h1.page-title{font-size:clamp(26px,7vw,34px);margin:10px 0 16px}.lede{font-size:16px}.section-head{margin:40px 0 12px}.proj-title{font-size:24px}.proj-meta-grid{padding:14px;gap:12px}.resume-grid{gap:6px}.resume-label{padding-top:16px;font-size:10px}.post-row{gap:8px;padding:14px 0}.post-row .when{font-size:10.5px}.post-row .title{font-size:16px}.post-row .read{display:none}.blog-controls{margin:4px 0 14px}}.route-fade{animation:fadein .26s cubic-bezier(.2,.6,.2,1)}@keyframes fadein{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}
