Portfolio — Classic One-column CV
A restrained, print-ready single-column CV for a single-person portfolio. Stacks a name-and-title header with a contact row, a serif summary, a timeline-style experience list, selected projects, education, and pill skills with language levels — all on a clean neutral base with a typographic rhythm tuned for ATS-friendly resumes. A floating toolbar prints or saves to PDF via @media print, swaps the accent color, and toggles compact spacing, with the chrome hidden on paper.
MCP
Code
:root {
--accent: #1f3a5f;
--ink: #14181f;
--ink-soft: #3a414c;
--ink-mute: #6b7280;
--rule: #e3e6eb;
--rule-soft: #eef0f3;
--paper: #ffffff;
--bg: #eceef1;
--max: 820px;
--pad-y: 56px;
--pad-x: 56px;
--gap: 30px;
--serif: "Source Serif 4", Georgia, "Times New Roman", serif;
--sans: "Inter", system-ui, -apple-system, "Segoe UI", Roboto, sans-serif;
--radius: 10px;
--shadow: 0 1px 2px rgba(16, 24, 40, 0.06), 0 18px 40px -18px rgba(16, 24, 40, 0.28);
}
* { box-sizing: border-box; }
html { -webkit-text-size-adjust: 100%; }
body {
margin: 0;
font-family: var(--sans);
color: var(--ink);
background: var(--bg);
line-height: 1.5;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
padding: 96px 18px 64px;
}
h1, h2, h3, p, ul, ol { margin: 0; }
ul, ol { padding: 0; list-style: none; }
a { color: inherit; }
/* ---------------- Toolbar ---------------- */
.toolbar {
position: fixed;
top: 16px;
left: 50%;
transform: translateX(-50%);
z-index: 30;
display: flex;
align-items: center;
gap: 18px;
flex-wrap: wrap;
justify-content: center;
padding: 9px 14px;
background: rgba(255, 255, 255, 0.86);
backdrop-filter: blur(10px);
-webkit-backdrop-filter: blur(10px);
border: 1px solid var(--rule);
border-radius: 999px;
box-shadow: 0 10px 30px -12px rgba(16, 24, 40, 0.3);
}
.toolbar__group {
display: flex;
align-items: center;
gap: 10px;
}
.toolbar__label {
font-size: 12px;
font-weight: 600;
letter-spacing: 0.06em;
text-transform: uppercase;
color: var(--ink-mute);
}
.swatches { display: flex; gap: 7px; }
.swatch {
width: 22px;
height: 22px;
border-radius: 50%;
border: 2px solid #fff;
background: var(--sw);
cursor: pointer;
padding: 0;
box-shadow: 0 0 0 1px var(--rule);
transition: transform 0.12s ease, box-shadow 0.12s ease;
}
.swatch:hover { transform: scale(1.12); }
.swatch[aria-checked="true"] {
box-shadow: 0 0 0 2px var(--paper), 0 0 0 4px var(--sw);
}
.btn {
font-family: var(--sans);
font-size: 13.5px;
font-weight: 600;
border-radius: 999px;
border: 1px solid transparent;
padding: 8px 15px;
cursor: pointer;
display: inline-flex;
align-items: center;
gap: 7px;
transition: background 0.15s ease, color 0.15s ease, border-color 0.15s ease, transform 0.08s ease;
}
.btn:active { transform: translateY(1px); }
.btn--ghost {
background: #fff;
color: var(--ink-soft);
border-color: var(--rule);
}
.btn--ghost:hover { background: #f5f6f8; }
.btn--ghost[aria-pressed="true"] {
background: var(--accent);
color: #fff;
border-color: var(--accent);
}
.btn--primary {
background: var(--accent);
color: #fff;
}
.btn--primary:hover { filter: brightness(1.08); }
:focus-visible {
outline: 3px solid color-mix(in srgb, var(--accent) 55%, #fff);
outline-offset: 2px;
}
/* ---------------- Page / paper ---------------- */
.page {
max-width: var(--max);
margin: 0 auto;
background: var(--paper);
border-radius: var(--radius);
box-shadow: var(--shadow);
padding: var(--pad-y) var(--pad-x);
position: relative;
}
.page::before {
content: "";
position: absolute;
inset: 0 auto 0 0;
width: 5px;
border-radius: var(--radius) 0 0 var(--radius);
background: var(--accent);
}
/* ---------------- Header ---------------- */
.cv-head {
display: flex;
justify-content: space-between;
align-items: flex-end;
gap: 28px;
flex-wrap: wrap;
padding-bottom: 22px;
border-bottom: 2px solid var(--accent);
}
.cv-name {
font-family: var(--serif);
font-weight: 700;
font-size: clamp(30px, 5vw, 42px);
line-height: 1.05;
letter-spacing: -0.015em;
color: var(--ink);
}
.cv-title {
margin-top: 6px;
font-size: 14.5px;
font-weight: 600;
letter-spacing: 0.01em;
color: var(--accent);
}
.cv-contact {
display: grid;
grid-template-columns: auto auto;
gap: 4px 20px;
font-size: 12.8px;
text-align: right;
}
.cv-contact li {
display: flex;
gap: 8px;
justify-content: flex-end;
align-items: baseline;
}
.cv-contact__k {
font-size: 10.5px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.08em;
color: var(--ink-mute);
}
.cv-contact a {
text-decoration: none;
color: var(--ink-soft);
border-bottom: 1px solid transparent;
transition: border-color 0.15s ease, color 0.15s ease;
}
.cv-contact a:hover {
color: var(--accent);
border-bottom-color: var(--accent);
}
/* ---------------- Sections ---------------- */
.cv-section { margin-top: var(--gap); }
.cv-section__title {
font-family: var(--sans);
font-size: 12px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.16em;
color: var(--accent);
padding-bottom: 6px;
margin-bottom: 14px;
border-bottom: 1px solid var(--rule);
}
.cv-summary {
font-family: var(--serif);
font-size: 16px;
line-height: 1.6;
color: var(--ink-soft);
max-width: 64ch;
}
/* ---------------- Timeline / entries ---------------- */
.timeline { position: relative; }
.entry { padding: 0 0 18px 22px; position: relative; }
.entry:last-child { padding-bottom: 0; }
.entry::before {
content: "";
position: absolute;
left: 4px;
top: 7px;
bottom: -2px;
width: 2px;
background: var(--rule);
}
.entry:last-child::before { bottom: 12px; }
.entry::after {
content: "";
position: absolute;
left: 0;
top: 5px;
width: 10px;
height: 10px;
border-radius: 50%;
background: var(--paper);
border: 2.5px solid var(--accent);
}
.entry--tight { padding-left: 22px; padding-bottom: 12px; }
.entry__head {
display: flex;
justify-content: space-between;
align-items: baseline;
gap: 16px;
flex-wrap: wrap;
}
.entry__role {
font-family: var(--serif);
font-size: 17px;
font-weight: 600;
color: var(--ink);
}
.entry__org {
font-size: 13.5px;
font-weight: 500;
color: var(--ink-mute);
margin-top: 1px;
}
.entry__meta {
font-size: 12px;
font-weight: 600;
letter-spacing: 0.04em;
color: var(--ink-mute);
white-space: nowrap;
font-variant-numeric: tabular-nums;
}
.entry__points {
margin-top: 9px;
display: grid;
gap: 5px;
}
.entry__points li {
position: relative;
padding-left: 16px;
font-size: 13.8px;
line-height: 1.5;
color: var(--ink-soft);
}
.entry__points li::before {
content: "";
position: absolute;
left: 0;
top: 9px;
width: 5px;
height: 5px;
border-radius: 50%;
background: var(--accent);
}
.entry__points em {
font-style: italic;
color: var(--ink);
font-weight: 600;
}
/* ---------------- Projects ---------------- */
.projects {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 12px 24px;
}
.project {
padding: 12px 14px;
border: 1px solid var(--rule);
border-radius: 8px;
border-left: 3px solid var(--accent);
background: linear-gradient(180deg, #fff, #fafbfc);
transition: box-shadow 0.15s ease, transform 0.15s ease;
}
.project:hover {
box-shadow: 0 8px 20px -12px rgba(16, 24, 40, 0.35);
transform: translateY(-2px);
}
.project__name {
font-family: var(--serif);
font-size: 15px;
font-weight: 600;
color: var(--ink);
}
.project__desc {
margin-top: 4px;
font-size: 12.8px;
line-height: 1.45;
color: var(--ink-mute);
}
/* ---------------- Lower grid ---------------- */
.cv-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: var(--gap);
margin-top: var(--gap);
}
.cv-grid .cv-section { margin-top: 0; }
.skills {
display: flex;
flex-wrap: wrap;
gap: 7px;
}
.skills li {
font-size: 12.5px;
font-weight: 500;
padding: 4px 11px;
border-radius: 999px;
background: color-mix(in srgb, var(--accent) 8%, #fff);
color: var(--accent);
border: 1px solid color-mix(in srgb, var(--accent) 22%, #fff);
}
.cv-subhead {
font-size: 11px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.1em;
color: var(--ink-mute);
margin: 16px 0 8px;
}
.langs { display: grid; gap: 5px; }
.langs li {
display: flex;
justify-content: space-between;
font-size: 13px;
color: var(--ink-soft);
padding-bottom: 4px;
border-bottom: 1px dotted var(--rule);
}
.langs__lvl { color: var(--ink-mute); font-weight: 500; }
/* ---------------- Footer ---------------- */
.cv-foot {
display: flex;
justify-content: space-between;
flex-wrap: wrap;
gap: 8px;
margin-top: var(--gap);
padding-top: 16px;
border-top: 1px solid var(--rule);
font-size: 11.5px;
letter-spacing: 0.04em;
color: var(--ink-mute);
}
/* ---------------- Compact density ---------------- */
body.is-compact {
--gap: 20px;
--pad-y: 40px;
}
body.is-compact .cv-summary { font-size: 14.5px; }
body.is-compact .entry { padding-bottom: 12px; }
body.is-compact .entry__points { margin-top: 6px; gap: 3px; }
body.is-compact .entry__points li { font-size: 13px; }
/* ---------------- Toast ---------------- */
.toast {
position: fixed;
bottom: 26px;
left: 50%;
transform: translate(-50%, 16px);
background: var(--ink);
color: #fff;
font-size: 13.5px;
font-weight: 500;
padding: 11px 18px;
border-radius: 10px;
box-shadow: 0 12px 30px -10px rgba(0, 0, 0, 0.5);
opacity: 0;
pointer-events: none;
transition: opacity 0.2s ease, transform 0.2s ease;
z-index: 50;
}
.toast.is-show {
opacity: 1;
transform: translate(-50%, 0);
}
/* ---------------- Responsive ---------------- */
@media (max-width: 720px) {
:root { --pad-x: 30px; --pad-y: 38px; }
body { padding-top: 120px; }
.cv-head { align-items: flex-start; }
.cv-contact {
grid-template-columns: 1fr;
text-align: left;
gap: 3px;
}
.cv-contact li { justify-content: flex-start; }
.projects, .cv-grid { grid-template-columns: 1fr; }
}
@media (max-width: 420px) {
:root { --pad-x: 20px; }
body { padding: 132px 10px 40px; }
.entry__head { flex-direction: column; gap: 2px; }
.entry__meta { order: -1; }
.toolbar { gap: 12px; padding: 8px 12px; }
.toolbar__label { display: none; }
}
/* ---------------- Print ---------------- */
@media print {
@page { size: A4; margin: 16mm 15mm; }
body {
background: #fff;
padding: 0;
color: #000;
}
.toolbar, .toast { display: none !important; }
.page {
max-width: none;
box-shadow: none;
border-radius: 0;
padding: 0;
}
.page::before { display: none; }
a { color: #000; text-decoration: none; }
.cv-contact a { border-bottom: none; }
.project:hover { box-shadow: none; transform: none; }
.cv-head { border-bottom-color: #000; }
.entry, .project, .cv-section { break-inside: avoid; }
.cv-section { break-inside: avoid-page; }
}(function () {
"use strict";
var root = document.documentElement;
var body = document.body;
/* ---------- Toast helper ---------- */
var toastEl = document.getElementById("toast");
var toastTimer = null;
function toast(msg) {
if (!toastEl) return;
toastEl.textContent = msg;
toastEl.classList.add("is-show");
window.clearTimeout(toastTimer);
toastTimer = window.setTimeout(function () {
toastEl.classList.remove("is-show");
}, 2200);
}
/* ---------- Accent color picker ---------- */
var swatches = Array.prototype.slice.call(document.querySelectorAll(".swatch"));
var ACCENT_KEY = "cv-accent";
function applyAccent(color, announce) {
root.style.setProperty("--accent", color);
swatches.forEach(function (s) {
var active = s.dataset.accent === color;
s.setAttribute("aria-checked", active ? "true" : "false");
s.tabIndex = active ? 0 : -1;
});
try { localStorage.setItem(ACCENT_KEY, color); } catch (e) {}
if (announce) {
var name = announce.getAttribute("aria-label") || "accent";
toast("Accent set to " + name);
}
}
swatches.forEach(function (s, i) {
s.addEventListener("click", function () {
applyAccent(s.dataset.accent, s);
});
// Roving keyboard support inside the radiogroup
s.addEventListener("keydown", function (e) {
var dir = 0;
if (e.key === "ArrowRight" || e.key === "ArrowDown") dir = 1;
else if (e.key === "ArrowLeft" || e.key === "ArrowUp") dir = -1;
else return;
e.preventDefault();
var next = (i + dir + swatches.length) % swatches.length;
swatches[next].focus();
applyAccent(swatches[next].dataset.accent, swatches[next]);
});
});
// Restore a previously chosen accent
try {
var saved = localStorage.getItem(ACCENT_KEY);
if (saved && swatches.some(function (s) { return s.dataset.accent === saved; })) {
applyAccent(saved, null);
}
} catch (e) {}
/* ---------- Density toggle ---------- */
var densityBtn = document.getElementById("density-btn");
if (densityBtn) {
densityBtn.addEventListener("click", function () {
var compact = body.classList.toggle("is-compact");
densityBtn.setAttribute("aria-pressed", compact ? "true" : "false");
densityBtn.textContent = compact ? "Comfortable" : "Compact";
toast(compact ? "Compact spacing on" : "Comfortable spacing on");
});
}
/* ---------- Print / Save PDF ---------- */
var printBtn = document.getElementById("print-btn");
if (printBtn) {
printBtn.addEventListener("click", function () {
toast("Opening print dialog — choose “Save as PDF”");
// Let the toast paint before the (blocking) print dialog opens.
window.setTimeout(function () { window.print(); }, 180);
});
}
// Ctrl/Cmd+P keeps working and feels native.
document.addEventListener("keydown", function (e) {
if ((e.ctrlKey || e.metaKey) && (e.key === "p" || e.key === "P")) {
// browser handles print itself; nothing to override
}
});
})();<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>Maya Okafor — Product Designer · CV</title>
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link href="https://fonts.googleapis.com/css2?family=Source+Serif+4:opsz,[email protected],400;8..60,600;8..60,700&family=Inter:wght@400;500;600;700&display=swap" rel="stylesheet" />
<link rel="stylesheet" href="style.css" />
</head>
<body>
<!-- Floating control bar (hidden when printing) -->
<div class="toolbar" role="toolbar" aria-label="Document actions">
<div class="toolbar__group" role="group" aria-label="Accent color">
<span class="toolbar__label" id="accent-label">Accent</span>
<div class="swatches" role="radiogroup" aria-labelledby="accent-label">
<button class="swatch" type="button" role="radio" aria-checked="true" data-accent="#1f3a5f" style="--sw:#1f3a5f" aria-label="Navy"></button>
<button class="swatch" type="button" role="radio" aria-checked="false" data-accent="#0f766e" style="--sw:#0f766e" aria-label="Teal"></button>
<button class="swatch" type="button" role="radio" aria-checked="false" data-accent="#7c2d12" style="--sw:#7c2d12" aria-label="Rust"></button>
<button class="swatch" type="button" role="radio" aria-checked="false" data-accent="#6d28d9" style="--sw:#6d28d9" aria-label="Violet"></button>
<button class="swatch" type="button" role="radio" aria-checked="false" data-accent="#111827" style="--sw:#111827" aria-label="Ink"></button>
</div>
</div>
<div class="toolbar__group">
<button class="btn btn--ghost" type="button" id="density-btn" aria-pressed="false">Compact</button>
<button class="btn btn--primary" type="button" id="print-btn">
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><polyline points="6 9 6 2 18 2 18 9"></polyline><path d="M6 18H4a2 2 0 0 1-2-2v-5a2 2 0 0 1 2-2h16a2 2 0 0 1 2 2v5a2 2 0 0 1-2 2h-2"></path><rect x="6" y="14" width="12" height="8"></rect></svg>
Print / Save PDF
</button>
</div>
</div>
<main class="page" role="main" aria-label="Curriculum vitae">
<!-- Header -->
<header class="cv-head">
<div class="cv-head__id">
<h1 class="cv-name">Maya Okafor</h1>
<p class="cv-title">Senior Product Designer · Design Systems & 0→1</p>
</div>
<ul class="cv-contact" aria-label="Contact details">
<li><span class="cv-contact__k">Email</span><a href="mailto:[email protected]">[email protected]</a></li>
<li><span class="cv-contact__k">Site</span><a href="https://okafor.design" rel="noopener">okafor.design</a></li>
<li><span class="cv-contact__k">Based</span><span>Lisbon, PT · Remote</span></li>
<li><span class="cv-contact__k">Phone</span><a href="tel:+351910000000">+351 910 000 000</a></li>
</ul>
</header>
<!-- Summary -->
<section class="cv-section" aria-labelledby="sec-summary">
<h2 class="cv-section__title" id="sec-summary">Summary</h2>
<p class="cv-summary">
Product designer with 9 years shaping fintech, health, and developer tools from
first sketch to shipped system. I pair rigorous UX research with a craft-led eye
for typography and motion, and I love turning ambiguous problems into measurable
product wins. Comfortable owning end-to-end flows, scaling component libraries, and
mentoring designers toward sharper, calmer work.
</p>
</section>
<!-- Experience -->
<section class="cv-section" aria-labelledby="sec-exp">
<h2 class="cv-section__title" id="sec-exp">Experience</h2>
<ol class="timeline">
<li class="entry">
<div class="entry__head">
<div>
<h3 class="entry__role">Senior Product Designer</h3>
<p class="entry__org">Northwind Pay · Lisbon</p>
</div>
<p class="entry__meta"><time>2022</time> — <time>Present</time></p>
</div>
<ul class="entry__points">
<li>Led the redesign of the merchant dashboard, lifting weekly active use 38% and cutting support tickets by a third.</li>
<li>Built <em>Slate</em>, a 140-component design system adopted by four product squads, with tokenized theming and accessibility baked in.</li>
<li>Ran a quarterly research cadence that reframed onboarding, shrinking time-to-first-payout from 6 days to 2.</li>
</ul>
</li>
<li class="entry">
<div class="entry__head">
<div>
<h3 class="entry__role">Product Designer</h3>
<p class="entry__org">Cresset Health · Berlin</p>
</div>
<p class="entry__meta"><time>2019</time> — <time>2022</time></p>
</div>
<ul class="entry__points">
<li>Designed a clinician scheduling tool used by 1,200+ practices, balancing dense data with a calm, scannable layout.</li>
<li>Owned the mobile patient app from concept to launch; sustained a 4.7★ rating across 30k reviews.</li>
<li>Introduced design-ops rituals (critique, spec handoff, audit) that halved engineering rework.</li>
</ul>
</li>
<li class="entry">
<div class="entry__head">
<div>
<h3 class="entry__role">UX Designer</h3>
<p class="entry__org">Loom & Lever Studio · Remote</p>
</div>
<p class="entry__meta"><time>2016</time> — <time>2019</time></p>
</div>
<ul class="entry__points">
<li>Shipped 20+ client products across SaaS, retail, and civic tech as the studio's sole interaction designer.</li>
<li>Established the studio's reusable Figma kit, cutting average project kickoff time by two weeks.</li>
</ul>
</li>
</ol>
</section>
<!-- Selected work -->
<section class="cv-section" aria-labelledby="sec-work">
<h2 class="cv-section__title" id="sec-work">Selected Projects</h2>
<ul class="projects">
<li class="project">
<h3 class="project__name">Slate Design System</h3>
<p class="project__desc">Tokenized, themeable component library powering four Northwind products. 140 components, full a11y coverage.</p>
</li>
<li class="project">
<h3 class="project__name">Payout Onboarding</h3>
<p class="project__desc">Reworked merchant activation flow — reduced time-to-first-payout from six days to two.</p>
</li>
<li class="project">
<h3 class="project__name">Clinic Scheduler</h3>
<p class="project__desc">Dense-but-calm scheduling surface adopted by 1,200+ medical practices across the EU.</p>
</li>
<li class="project">
<h3 class="project__name">Field Notes (Side)</h3>
<p class="project__desc">A minimalist research-logging app, 5k installs, featured in two design newsletters.</p>
</li>
</ul>
</section>
<!-- Two-column lower band -->
<div class="cv-grid">
<!-- Education -->
<section class="cv-section" aria-labelledby="sec-edu">
<h2 class="cv-section__title" id="sec-edu">Education</h2>
<div class="entry entry--tight">
<div class="entry__head">
<div>
<h3 class="entry__role">MA, Interaction Design</h3>
<p class="entry__org">Royal College of Design</p>
</div>
<p class="entry__meta"><time>2016</time></p>
</div>
</div>
<div class="entry entry--tight">
<div class="entry__head">
<div>
<h3 class="entry__role">BSc, Cognitive Science</h3>
<p class="entry__org">University of Edinburgh</p>
</div>
<p class="entry__meta"><time>2014</time></p>
</div>
</div>
</section>
<!-- Skills -->
<section class="cv-section" aria-labelledby="sec-skills">
<h2 class="cv-section__title" id="sec-skills">Skills</h2>
<ul class="skills" aria-label="Core skills">
<li>Design Systems</li>
<li>UX Research</li>
<li>Prototyping</li>
<li>Figma</li>
<li>Accessibility (WCAG)</li>
<li>Interaction Design</li>
<li>Design Ops</li>
<li>HTML & CSS</li>
<li>Motion</li>
<li>Workshop Facilitation</li>
</ul>
<h3 class="cv-subhead">Languages</h3>
<ul class="langs">
<li><span>English</span><span class="langs__lvl">Native</span></li>
<li><span>Portuguese</span><span class="langs__lvl">Fluent</span></li>
<li><span>German</span><span class="langs__lvl">Conversational</span></li>
</ul>
</section>
</div>
<footer class="cv-foot">
<span>Maya Okafor — Curriculum Vitae</span>
<span>References available on request</span>
</footer>
</main>
<div class="toast" id="toast" role="status" aria-live="polite"></div>
<script src="script.js"></script>
</body>
</html>Classic One-column CV
A classic, single-column curriculum vitae rendered on a paper-like card. It opens with a name-and-title header and a two-up contact row, then flows through a serif summary, a timeline-style experience list with dotted markers, a grid of selected projects, and a lower band that pairs education with pill-style skills and language levels. Strong typographic hierarchy — a Source Serif headline voice over an Inter body — keeps it calm, professional, and ATS-friendly.
A floating toolbar carries the only interactions. The primary button triggers a
print-optimized PDF export: an @media print block hides the chrome, drops the
card shadow, sets A4 page margins, and keeps entries from breaking across pages.
A five-swatch accent picker recolors the rules, markers, and skill pills live and
remembers your choice in localStorage, with full arrow-key support inside the
radiogroup. A density toggle switches between comfortable and compact spacing for
fitting more onto a single page.
Everything is vanilla JS with no dependencies, and the layout collapses gracefully — the contact grid, project grid, and lower band all stack into a single column on small screens.
Illustrative portfolio — fictional person and projects.