Feat: Add Stain vs system font preview page

This commit is contained in:
2025-11-09 08:56:59 -08:00
parent 1187f635e9
commit b19dcf3751

View File

@@ -1,341 +1,188 @@
<!doctype html> <!DOCTYPE html>
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="utf-8"> <meta charset="UTF-8" />
<title>Stain vs Candara Preview</title> <title>Stain Font Preview</title>
<meta name="viewport" content="width=device-width,initial-scale=1"> <meta name="viewport" content="width=device-width, initial-scale=1" />
<style> <style>
:root { :root {
--bg: #f7f8fb; color-scheme: light;
--bg: #f7f7f8;
--fg: #111827; --fg: #111827;
--accent: #2563eb; --muted: #6b7280;
--grid: #e5e7eb; --accent: #111827;
--border: #d1d5db; --border: #e5e7eb;
--radius: 10px; --font-ui: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;
--font-ui: system-ui, -apple-system, BlinkMacSystemFont, -system-ui,
-apple-system,BlinkMacSystemFont,"Segoe UI",sans-serif;
} }
* { box-sizing: border-box; } * { box-sizing: border-box; }
body { body {
margin: 0; margin: 0;
padding: 24px; padding: 1.5rem;
font-family: var(--font-ui); font-family: var(--font-ui);
background: var(--bg); background: var(--bg);
color: var(--fg); color: var(--fg);
-webkit-font-smoothing: antialiased;
} }
h1 { h1 {
margin: 0 0 8px; margin: 0 0 1rem;
font-size: 1.8rem;
font-weight: 600; font-weight: 600;
letter-spacing: 0.02em; letter-spacing: 0.03em;
}
h2 {
margin: 24px 0 8px;
font-size: 16px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.16em;
color: #6b7280;
} }
p { p {
margin: 4px 0 16px; margin: 0 0 0.75rem;
color: #4b5563; color: var(--muted);
max-width: 720px; font-size: 0.95rem;
font-size: 14px;
}
.row {
display: flex;
flex-wrap: wrap;
gap: 16px;
margin-bottom: 16px;
}
.card {
flex: 1 1 260px;
background: #ffffff;
border-radius: var(--radius);
border: 1px solid var(--border);
padding: 14px 14px 12px;
box-shadow: 0 8px 18px rgba(15,23,42,0.04);
position: relative;
overflow: hidden;
}
.card-label {
font-size: 10px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.14em;
color: #9ca3af;
display: flex;
align-items: center;
gap: 6px;
margin-bottom: 4px;
}
.pill {
padding: 2px 8px 3px;
border-radius: 999px;
font-size: 9px;
font-weight: 600;
border: 1px solid #d1d5db;
color: #6b7280;
background: #f9fafb;
}
.sample {
font-size: 60px;
line-height: 1.1;
margin: 2px 0 6px;
display: flex;
flex-wrap: wrap;
gap: 10px 16px;
}
.sample span {
display: inline-block;
}
.caption {
font-size: 10px;
color: #9ca3af;
display: flex;
justify-content: space-between;
gap: 8px;
align-items: center;
}
.slider-wrap {
display: flex;
align-items: center;
gap: 8px;
font-size: 10px;
color: #6b7280;
}
input[type=range] {
-webkit-appearance: none;
appearance: none;
width: 120px;
height: 4px;
border-radius: 999px;
background: var(--grid);
outline: none;
}
input[type=range]::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
width: 13px;
height: 13px;
border-radius: 50%;
background: var(--accent);
cursor: pointer;
box-shadow: 0 1px 3px rgba(15,23,42,0.25);
}
input[type=range]::-moz-range-thumb {
width: 13px;
height: 13px;
border-radius: 50%;
background: var(--accent);
cursor: pointer;
box-shadow: 0 1px 3px rgba(15,23,42,0.25);
} }
.grid { .grid {
position: absolute; display: grid;
inset: 0; grid-template-columns: minmax(0, 1fr);
background-image: gap: 1.25rem;
linear-gradient(to bottom, rgba(229,231,235,0.6) 1px, transparent 1px), margin-top: 1.5rem;
linear-gradient(to right, rgba(229,231,235,0.25) 1px, transparent 1px);
background-size: 22px 22px, 24px 24px;
opacity: 0.18;
pointer-events: none;
mix-blend-mode: multiply;
} }
.grid-labels { @media (min-width: 768px) {
position: absolute; .grid {
left: 6px; grid-template-columns: repeat(2, minmax(0, 1fr));
top: 8px; }
font-size: 7px; }
color: rgba(148,163,253,0.7); .panel {
padding: 1rem 1.1rem;
border-radius: 0.85rem;
border: 1px solid var(--border);
background: #ffffff;
box-shadow: 0 10px 25px rgba(15,23,42,0.03);
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 4px; gap: 0.5rem;
pointer-events: none;
text-shadow: 0 1px 1px rgba(255,255,255,0.9);
} }
.grid-labels span { .label {
padding: 1px 4px; font-size: 0.8rem;
border-radius: 999px; text-transform: uppercase;
background: rgba(255,255,255,0.9); letter-spacing: 0.16em;
border: 1px solid rgba(199,210,254,0.8); color: var(--muted);
} }
.code { .sample {
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; min-height: 80px;
font-size: 9px; font-size: 40px;
padding: 3px 6px; letter-spacing: 0.03em;
border-radius: 6px;
background: #f9fafb;
border: 1px solid #e5e7eb;
color: #6b7280;
}
.flex-between {
display: flex; display: flex;
justify-content: space-between;
align-items: center; align-items: center;
gap: 8px; flex-wrap: wrap;
gap: 0.4ch;
} }
.small-link { .controls {
font-size: 9px; display: flex;
color: var(--accent); flex-wrap: wrap;
text-decoration: none; gap: 0.5rem;
align-items: center;
margin-top: 1rem;
font-size: 0.8rem;
color: var(--muted);
} }
.small-link:hover { .controls input[type="range"] {
text-decoration: underline; flex: 1;
} }
@media (max-width: 640px) { .input-wrap {
body { margin-top: 0.5rem;
padding: 16px; display: flex;
} align-items: center;
.sample { gap: 0.4rem;
font-size: 42px; font-size: 0.8rem;
} color: var(--muted);
}
.input-wrap input {
flex: 1;
padding: 0.35rem 0.5rem;
border-radius: 0.4rem;
border: 1px solid var(--border);
font-family: var(--font-ui);
font-size: 0.85rem;
}
.tag {
padding: 0.15rem 0.5rem;
border-radius: 999px;
border: 1px solid var(--border);
font-size: 0.7rem;
color: var(--muted);
} }
@font-face { @font-face {
font-family: "Stain"; font-family: "Stain";
src: url("./dist/Stain-Regular.ttf") format("truetype"); src: url("./dist/Stain.otf") format("opentype");
font-weight: 400; font-weight: 400;
font-style: normal; font-style: normal;
font-display: swap;
} }
.stain { .stain { font-family: "Stain", system-ui, sans-serif; }
font-family: "Stain", system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif; .sys {
font-feature-settings: "liga" 1, "kern" 1; font-family:
} "Candara",
.candara { system-ui,
font-family: Candara, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif; -apple-system,
BlinkMacSystemFont,
"Segoe UI",
sans-serif;
} }
</style> </style>
</head> </head>
<body> <body>
<header> <h1>Stain Font Preview</h1>
<h1>Stain Typeface Preview</h1> <p>
<p> Compare the in-progress "Stain" typeface against your system text (ideally Candara) in light mode.
Stain is a Candara-inspired humanist sans experiment. Only A, B, C / a, b, c are currently implemented.
This page compares Stain (left) with Candara (right) for A, B, C / a, b, c. </p>
Make sure Candara is installed locally to see it accurately.
</p>
</header>
<h2>Core glyphs: A B C / a b c</h2> <div class="controls">
<div class="row"> <span class="tag">A B C a b c</span>
<section class="card"> <span>Size</span>
<div class="grid" aria-hidden="true"></div> <input id="size" type="range" min="18" max="96" value="40" />
<div class="grid-labels" aria-hidden="true"> <span id="sizeVal">40px</span>
<span>Ascender</span> <span>Letter spacing</span>
<span>x-height</span> <input id="track" type="range" min="-10" max="40" value="3" />
<span>Baseline</span> <span id="trackVal">0.03em</span>
</div>
<div class="card-label">
<span class="pill">Stain</span>
<span>Prototype outlines</span>
</div>
<div class="sample stain" id="stainSample">
<span>A</span><span>B</span><span>C</span>
<span>a</span><span>b</span><span>c</span>
</div>
<div class="caption">
<div class="slider-wrap">
Size
<input id="stainSize" type="range" min="28" max="96" value="60">
</div>
<code class="code">font-family: "Stain"</code>
</div>
</section>
<section class="card">
<div class="card-label">
<span class="pill">Candara</span>
<span>Reference glyphs</span>
</div>
<div class="sample candara" id="candaraSample">
<span>A</span><span>B</span><span>C</span>
<span>a</span><span>b</span><span>c</span>
</div>
<div class="caption flex-between">
<div class="slider-wrap">
Size
<input id="candaraSize" type="range" min="28" max="96" value="60">
</div>
<div>
<span class="code">font-family: Candara</span>
</div>
</div>
</section>
</div> </div>
<h2>Side-by-side comparison</h2> <div class="input-wrap">
<div class="row"> <span>Sample text:</span>
<section class="card"> <input id="text" value="ABC abc" />
<div class="card-label"> </div>
<span class="pill">Pair</span>
Overlay scan-lines <div class="grid">
</div> <section class="panel">
<div class="sample"> <div class="label">Stain</div>
<span class="stain" id="pairStain">ABC abc</span> <div id="stainSample" class="sample stain">ABC abc</div>
<span class="candara" id="pairCandara">ABC abc</span>
</div>
<div class="caption flex-between">
<div class="slider-wrap">
Shared size
<input id="pairSize" type="range" min="28" max="96" value="54">
</div>
<span class="code">Compare curves / modulation / width</span>
</div>
</section> </section>
<section class="card"> <section class="panel">
<div class="card-label"> <div class="label">System (Candara / fallback)</div>
<span class="pill">Playground</span> <div id="sysSample" class="sample sys">ABC abc</div>
Custom text
</div>
<div class="sample" style="flex-direction: column; gap: 6px;">
<div class="stain" id="playStain">Stain ABC abc</div>
<div class="candara" id="playCandara">Stain ABC abc</div>
</div>
<div class="caption flex-between">
<input
id="playInput"
type="text"
value="Stain ABC abc"
style="flex:1;border:1px solid var(--border);border-radius:7px;padding:4px 6px;font-size:10px;font-family:var(--font-ui);color:#374151;background:#f9fafb;"
>
<a class="small-link" href="./dist/Stain-Regular.ttf" download>Download Stain-Regular.ttf</a>
</div>
</section> </section>
</div> </div>
<script> <script>
const text = document.getElementById("text");
const size = document.getElementById("size");
const sizeVal = document.getElementById("sizeVal");
const track = document.getElementById("track");
const trackVal = document.getElementById("trackVal");
const stainSample = document.getElementById("stainSample"); const stainSample = document.getElementById("stainSample");
const candaraSample = document.getElementById("candaraSample"); const sysSample = document.getElementById("sysSample");
const pairStain = document.getElementById("pairStain");
const pairCandara = document.getElementById("pairCandara");
const playStain = document.getElementById("playStain");
const playCandara = document.getElementById("playCandara");
const stainSize = document.getElementById("stainSize"); const update = () => {
const candaraSize = document.getElementById("candaraSize"); const v = text.value || "ABC abc";
const pairSize = document.getElementById("pairSize"); const fs = size.valueAsNumber || 40;
const playInput = document.getElementById("playInput"); const ls = (track.valueAsNumber || 0) / 100;
sizeVal.textContent = fs + "px";
const sync = (input, els) => { trackVal.textContent = ls.toFixed(2) + "em";
const apply = () => { [stainSample, sysSample].forEach(el => {
const v = input.value; el.textContent = v;
els.forEach(e => e.style.fontSize = v + "px"); el.style.fontSize = fs + "px";
}; el.style.letterSpacing = ls + "em";
input.addEventListener("input", apply); });
apply();
}; };
sync(stainSize, [stainSample]); text.addEventListener("input", update);
sync(candaraSize, [candaraSample]); size.addEventListener("input", update);
sync(pairSize, [pairStain, pairCandara, playStain, playCandara]); track.addEventListener("input", update);
update();
playInput.addEventListener("input", () => {
const v = playInput.value || "Stain ABC abc";
playStain.textContent = v;
playCandara.textContent = v;
});
</script> </script>
</body> </body>
</html> </html>