Files
tsuki/assets/css/layout.css
T
tiennm99 d1828c9590 feat: v0.3.0 — Tier A parity, per-kind CSS, llm.txt, a11y baseline
Headline outcomes:
- Feature parity with Stack/PaperMod on Tier A polish: breadcrumbs
  (+BreadcrumbList JSON-LD), prev/next post navigation (rel=prev/next),
  language switcher UI (gated on hugo.IsMultilingual), code-copy button
  polish, <details> styling.
- Per-page-kind CSS bundles: core loads everywhere; home/single/archive/
  search bundles load only where needed. Frees ~1 KB gz from non-post
  pages. code-copy.js gated to post pages.
- Lighthouse-relevant network polish: Pagefind UI CSS preload-swap,
  conditional preconnect to giscus.app on comment-enabled posts,
  hreflang alternates on multilingual sites, <meta name=theme-color>
  light/dark variants, aria-pressed SSR-rendered on theme-toggle.
- WCAG AA contrast: --tsuki-fg-subtle darkened to #6b6b6b (light); was
  3.54:1 on bg, now 5:1. Pagination disabled uses --tsuki-fg-muted
  without opacity compound. Header tap targets bumped to 40×40; pagination
  to 44×44.
- i18n: full i18n/en.yml mirror (~50 keys); render-heading aria-label
  i18n-driven via linkToSection key; new keys for breadcrumb*, prevPost,
  nextPost, copyCode, copiedCode.
- Discovery: /llm.txt output format (llmstxt.org) on home; Speculation
  Rules opt-in via params.prefetch.enable.
- <html lang> fallback: site.Language.LanguageCode → Lang → "en"
  (was hard-coded "vi").

Per-kind bundle gz sizes on demo: home 3673 / post 4167 / list 2897 /
archives 3363 / search 3179 B (all under 4200 B cap).

Plan: plans/260510-0144-tsuki-v0.3.0/
2026-05-15 18:55:47 +07:00

140 lines
3.2 KiB
CSS

/* tsuki: page layout — header, main, footer composition */
body {
display: grid;
grid-template-rows: auto 1fr auto;
}
.site-header,
main,
.site-footer { padding-inline: var(--tsuki-space-6); }
.site-header {
display: flex;
align-items: center;
gap: var(--tsuki-space-4);
flex-wrap: wrap;
max-width: var(--tsuki-wide-width);
margin-inline: auto;
width: 100%;
padding-block: var(--tsuki-space-6);
border-bottom: 1px solid var(--tsuki-border);
}
.site-title {
font-weight: 650;
font-size: var(--tsuki-fs-lg);
letter-spacing: -0.01em;
color: var(--tsuki-fg);
text-decoration: none;
}
.site-title:hover { text-decoration: none; color: var(--tsuki-accent); }
.site-nav { margin-inline-start: auto; }
.site-nav ul {
display: flex;
gap: var(--tsuki-space-4);
list-style: none;
padding: 0;
margin: 0;
}
.site-nav a { color: var(--tsuki-fg-muted); }
.site-nav a:hover { color: var(--tsuki-fg); }
.site-nav a[aria-current="page"] { color: var(--tsuki-fg); font-weight: 550; }
.lang-switcher {
display: flex;
gap: var(--tsuki-space-2);
list-style: none;
padding: 0;
margin: 0;
font-size: var(--tsuki-fs-sm);
}
.lang-switcher a {
color: var(--tsuki-fg-muted);
padding: var(--tsuki-space-1) var(--tsuki-space-2);
}
.lang-switcher a:hover { color: var(--tsuki-fg); }
.lang-switcher a[aria-current="page"] {
color: var(--tsuki-fg);
font-weight: 550;
text-decoration: none;
}
.site-header-actions {
display: flex;
align-items: center;
gap: var(--tsuki-space-2);
margin-inline-start: var(--tsuki-space-3);
}
.search-button {
display: inline-grid;
place-items: center;
min-width: 2.5rem;
min-height: 2.5rem;
border: 1px solid var(--tsuki-border);
border-radius: 999px;
color: var(--tsuki-fg-muted);
transition: var(--tsuki-transition);
}
.search-button:hover {
color: var(--tsuki-fg);
border-color: var(--tsuki-fg-muted);
text-decoration: none;
}
.search-button svg { width: 1rem; height: 1rem; }
.theme-toggle {
background: transparent;
border: 1px solid var(--tsuki-border);
border-radius: 999px;
min-width: 2.5rem;
min-height: 2.5rem;
display: inline-grid;
place-items: center;
color: var(--tsuki-fg-muted);
transition: var(--tsuki-transition);
}
.theme-toggle:hover { color: var(--tsuki-fg); border-color: var(--tsuki-fg-muted); }
main {
max-width: var(--tsuki-content-width);
margin-inline: auto;
width: 100%;
padding-block: var(--tsuki-space-12) var(--tsuki-space-16);
}
body.home main,
body.list main { max-width: var(--tsuki-wide-width); }
.site-footer {
max-width: var(--tsuki-wide-width);
margin-inline: auto;
width: 100%;
padding-block: var(--tsuki-space-8);
border-top: 1px solid var(--tsuki-border);
color: var(--tsuki-fg-muted);
font-size: var(--tsuki-fs-sm);
display: flex;
justify-content: space-between;
flex-wrap: wrap;
gap: var(--tsuki-space-3);
}
.site-footer p { margin: 0; }
.site-footer a { display: inline-block; padding-block: var(--tsuki-space-1); }
.list-header {
margin-block-end: var(--tsuki-space-8);
border-block-end: 1px solid var(--tsuki-border);
padding-block-end: var(--tsuki-space-6);
}
.list-title {
font-size: var(--tsuki-fs-2xl);
margin: 0;
}
.list-description {
margin: var(--tsuki-space-2) 0 0;
color: var(--tsuki-fg-muted);
}