SaaS — Features / Product Tour
A benefit-led SaaS features and product-tour page with alternating copy plus inline-SVG product mockups, a sticky scrollspy sub-nav that tracks the active feature, an accessible tab control that swaps an automation rule-builder preview, a before-and-after comparison band, a working light and dark theme toggle, and an email-validated call-to-action — all self-contained vanilla HTML, CSS, and JavaScript with no images, frameworks, or backend.
MCP
Code
:root {
--bg: #f7f8fb;
--surface: #ffffff;
--surface-2: #f1f2f8;
--ink: #0f1222;
--muted: #646b85;
--brand: #6366f1;
--brand-d: #4f46e5;
--ok: #16a34a;
--warn: #d97706;
--danger: #dc2626;
--line: rgba(15, 18, 34, .1);
--line-2: rgba(15, 18, 34, .06);
--shadow: 0 1px 2px rgba(15,18,34,.06), 0 12px 30px -16px rgba(15,18,34,.22);
--shadow-sm: 0 1px 2px rgba(15,18,34,.06), 0 4px 14px -8px rgba(15,18,34,.18);
--radius: 16px;
--maxw: 1140px;
--font: "Inter", system-ui, -apple-system, Segoe UI, Roboto, sans-serif;
}
:root[data-theme="dark"] {
--bg: #0b0d18;
--surface: #14172a;
--surface-2: #1b1f37;
--ink: #eef0f8;
--muted: #9aa2bd;
--line: rgba(255,255,255,.12);
--line-2: rgba(255,255,255,.07);
--shadow: 0 1px 2px rgba(0,0,0,.4), 0 18px 40px -18px rgba(0,0,0,.6);
--shadow-sm: 0 1px 2px rgba(0,0,0,.35), 0 6px 18px -10px rgba(0,0,0,.5);
}
* { box-sizing: border-box; }
html { scroll-behavior: smooth; scroll-padding-top: 84px; }
body {
margin: 0;
font-family: var(--font);
background: var(--bg);
color: var(--ink);
line-height: 1.55;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
transition: background .3s, color .3s;
}
@media (prefers-reduced-motion: reduce) {
html { scroll-behavior: auto; }
* { transition: none !important; animation: none !important; }
}
.wrap { max-width: var(--maxw); margin: 0 auto; padding: 0 20px; }
h1, h2 { letter-spacing: -.02em; line-height: 1.12; margin: 0; }
a { color: inherit; }
.sr-only {
position: absolute; width: 1px; height: 1px; padding: 0; margin: -1px;
overflow: hidden; clip: rect(0,0,0,0); white-space: nowrap; border: 0;
}
.skip-link {
position: absolute; left: 12px; top: -48px; z-index: 100;
background: var(--brand); color: #fff; padding: 10px 14px; border-radius: 10px;
transition: top .2s; text-decoration: none; font-weight: 600;
}
.skip-link:focus { top: 12px; }
:focus-visible { outline: 3px solid color-mix(in srgb, var(--brand) 55%, transparent); outline-offset: 2px; border-radius: 8px; }
/* Buttons */
.btn {
display: inline-flex; align-items: center; justify-content: center; gap: 8px;
font: inherit; font-weight: 600; cursor: pointer; text-decoration: none;
border-radius: 12px; padding: 9px 16px; border: 1px solid transparent;
transition: transform .12s ease, background .2s, border-color .2s, box-shadow .2s;
white-space: nowrap;
}
.btn:active { transform: translateY(1px); }
.btn-lg { padding: 13px 22px; font-size: 1.02rem; border-radius: 14px; }
.btn-primary { background: var(--brand); color: #fff; box-shadow: 0 6px 18px -8px var(--brand); }
.btn-primary:hover { background: var(--brand-d); }
.btn-ghost { background: var(--surface); color: var(--ink); border-color: var(--line); }
.btn-ghost:hover { background: var(--surface-2); }
/* Topbar */
.topbar {
position: sticky; top: 0; z-index: 40;
background: color-mix(in srgb, var(--surface) 82%, transparent);
backdrop-filter: saturate(160%) blur(12px);
border-bottom: 1px solid var(--line);
}
.topbar-inner { display: flex; align-items: center; gap: 20px; height: 64px; }
.brand { display: inline-flex; align-items: center; gap: 9px; text-decoration: none; font-weight: 800; }
.brand-mark {
display: grid; place-items: center; width: 32px; height: 32px; border-radius: 9px;
background: linear-gradient(140deg, var(--brand), var(--brand-d)); color: #fff;
}
.brand-name { font-size: 1.12rem; letter-spacing: -.02em; }
.topnav { display: flex; gap: 4px; margin-left: 8px; }
.topnav a {
text-decoration: none; color: var(--muted); font-weight: 500; font-size: .94rem;
padding: 8px 12px; border-radius: 10px; transition: color .2s, background .2s;
}
.topnav a:hover { color: var(--ink); background: var(--surface-2); }
.topbar-actions { margin-left: auto; display: flex; align-items: center; gap: 10px; }
#themeToggle .theme-label { font-size: .9rem; }
/* Hero */
.hero { padding: 64px 0 44px; text-align: center; }
.hero-inner { max-width: 760px; }
.eyebrow {
display: inline-block; margin: 0 0 14px; font-weight: 600; font-size: .82rem;
letter-spacing: .08em; text-transform: uppercase; color: var(--brand-d);
background: color-mix(in srgb, var(--brand) 12%, transparent);
padding: 5px 12px; border-radius: 999px;
}
.hero h1 { font-size: clamp(2rem, 5vw, 3.2rem); font-weight: 800; }
.lede { margin: 18px auto 0; max-width: 620px; color: var(--muted); font-size: clamp(1.02rem, 2.2vw, 1.18rem); }
.hero-cta { display: flex; gap: 12px; justify-content: center; flex-wrap: wrap; margin-top: 28px; }
.trust-row {
list-style: none; display: flex; gap: 14px; justify-content: center; flex-wrap: wrap;
margin: 26px 0 0; padding: 0; color: var(--muted); font-size: .9rem; font-weight: 500;
}
/* Tour layout */
.tour-layout { display: grid; grid-template-columns: 220px 1fr; gap: 40px; padding-top: 28px; padding-bottom: 40px; align-items: start; }
.spy { position: sticky; top: 84px; }
.spy-title { font-size: .76rem; text-transform: uppercase; letter-spacing: .08em; color: var(--muted); margin: 0 0 12px; font-weight: 600; }
.spy-list { list-style: none; margin: 0; padding: 0; border-left: 2px solid var(--line); }
.spy-list a {
display: block; text-decoration: none; color: var(--muted); font-weight: 500; font-size: .94rem;
padding: 8px 0 8px 16px; margin-left: -2px; border-left: 2px solid transparent;
transition: color .2s, border-color .2s;
}
.spy-list a:hover { color: var(--ink); }
.spy-list a.is-active { color: var(--brand-d); border-left-color: var(--brand); font-weight: 600; }
.features { display: flex; flex-direction: column; gap: 88px; min-width: 0; }
/* Feature blocks */
.feature {
display: grid; grid-template-columns: 1fr 1fr; gap: 48px; align-items: center;
scroll-margin-top: 84px;
}
.feature-rev .feature-copy { order: 2; }
.feature-stack { grid-template-columns: 1fr 1fr; }
.kicker {
display: inline-block; font-weight: 700; font-size: .76rem; letter-spacing: .07em;
text-transform: uppercase; padding: 4px 10px; border-radius: 999px; margin-bottom: 14px;
}
.kicker-1 { color: #2563eb; background: rgba(37,99,235,.12); }
.kicker-2 { color: #0891b2; background: rgba(8,145,178,.12); }
.kicker-3 { color: var(--brand-d); background: color-mix(in srgb, var(--brand) 14%, transparent); }
.feature-copy h2 { font-size: clamp(1.5rem, 3.4vw, 2.1rem); font-weight: 800; }
.feature-copy > p { color: var(--muted); margin: 14px 0 0; font-size: 1.04rem; }
.bullets { list-style: none; margin: 18px 0 0; padding: 0; display: grid; gap: 10px; }
.bullets li { position: relative; padding-left: 28px; color: var(--ink); font-weight: 500; }
.bullets li::before {
content: "✓"; position: absolute; left: 0; top: -1px; width: 19px; height: 19px;
display: grid; place-items: center; font-size: .72rem; font-weight: 800; color: #fff;
background: var(--ok); border-radius: 6px;
}
.link-arrow {
display: inline-flex; align-items: center; gap: 6px; margin-top: 22px;
color: var(--brand-d); font-weight: 600; text-decoration: none;
}
.link-arrow span { transition: transform .2s; }
.link-arrow:hover span { transform: translateX(4px); }
/* Mock shells */
.feature-mock { min-width: 0; }
.mock {
background: var(--surface); border: 1px solid var(--line); border-radius: var(--radius);
box-shadow: var(--shadow); overflow: hidden;
}
.mock-head {
display: flex; align-items: center; gap: 7px; padding: 11px 14px;
border-bottom: 1px solid var(--line-2); background: var(--surface-2);
}
.dot { width: 11px; height: 11px; border-radius: 50%; }
.dot.r { background: #ff5f57; } .dot.y { background: #febc2e; } .dot.g { background: #28c840; }
.mock-title { margin-left: 8px; font-size: .84rem; font-weight: 600; color: var(--muted); }
/* Board mock */
.board-cols { display: grid; grid-template-columns: repeat(3, 1fr); gap: 10px; padding: 14px; }
.bcol-h { display: flex; justify-content: space-between; font-size: .76rem; font-weight: 700; color: var(--muted); margin: 0 0 9px; text-transform: uppercase; letter-spacing: .04em; }
.bcol-h span { background: var(--surface-2); border: 1px solid var(--line-2); border-radius: 999px; padding: 0 7px; font-weight: 700; color: var(--ink); }
.card {
background: var(--surface); border: 1px solid var(--line); border-radius: 11px;
padding: 10px; margin-bottom: 9px; box-shadow: var(--shadow-sm); position: relative;
transition: transform .15s, box-shadow .15s;
}
.card:hover { transform: translateY(-2px); box-shadow: var(--shadow); }
.card p { margin: 7px 0 0; font-size: .82rem; font-weight: 600; line-height: 1.35; }
.card.hot { border-color: color-mix(in srgb, var(--warn) 50%, var(--line)); }
.card.done { opacity: .72; }
.tag { display: inline-block; font-size: .64rem; font-weight: 700; padding: 2px 8px; border-radius: 999px; letter-spacing: .03em; }
.tag-blue { color: #2563eb; background: rgba(37,99,235,.14); }
.tag-violet { color: var(--brand-d); background: color-mix(in srgb, var(--brand) 16%, transparent); }
.tag-amber { color: var(--warn); background: rgba(217,119,6,.14); }
.tag-green { color: var(--ok); background: rgba(22,163,74,.14); }
.bar { height: 5px; background: var(--surface-2); border-radius: 999px; margin-top: 9px; overflow: hidden; }
.bar i { display: block; height: 100%; background: var(--warn); border-radius: 999px; }
.avatar { width: 22px; height: 22px; border-radius: 50%; margin-top: 9px; border: 2px solid var(--surface); }
.a1 { background: linear-gradient(135deg, #6366f1, #a855f7); }
.a2 { background: linear-gradient(135deg, #0891b2, #22d3ee); }
.a3 { background: linear-gradient(135deg, #f59e0b, #ef4444); }
/* Chart mock */
.chart-stats { display: flex; gap: 12px; padding: 16px 16px 4px; flex-wrap: wrap; }
.cs-k { margin: 0; font-size: .72rem; text-transform: uppercase; letter-spacing: .05em; color: var(--muted); font-weight: 600; }
.cs-v { margin: 3px 0 0; font-size: 1.3rem; font-weight: 800; }
.cs-v em { font-size: .76rem; font-weight: 700; font-style: normal; vertical-align: middle; }
.up { color: var(--ok); } .down { color: var(--ok); }
.spark { width: 100%; height: 130px; display: block; padding: 8px 8px 0; }
/* Flow / automation mock */
.flow { padding: 18px 16px 22px; display: grid; gap: 10px; }
.flow-node {
display: flex; align-items: center; gap: 11px; padding: 12px 14px;
border: 1px solid var(--line); border-radius: 12px; background: var(--surface-2);
font-size: .88rem; font-weight: 600; box-shadow: var(--shadow-sm);
animation: fadein .35s ease both;
}
.flow-node .fn-ico { width: 30px; height: 30px; border-radius: 9px; display: grid; place-items: center; font-size: 1rem; flex-shrink: 0; }
.flow-node small { display: block; font-weight: 500; color: var(--muted); font-size: .76rem; }
.fn-trigger .fn-ico { background: rgba(37,99,235,.14); }
.fn-cond .fn-ico { background: color-mix(in srgb, var(--brand) 16%, transparent); }
.fn-action .fn-ico { background: rgba(22,163,74,.14); }
.flow-arrow { text-align: center; color: var(--muted); font-size: .9rem; line-height: 1; }
@keyframes fadein { from { opacity: 0; transform: translateY(6px); } to { opacity: 1; transform: none; } }
/* Tabs */
.tabs { display: flex; gap: 6px; margin-top: 20px; flex-wrap: wrap; background: var(--surface-2); padding: 5px; border-radius: 12px; border: 1px solid var(--line-2); width: fit-content; }
.tab {
font: inherit; font-weight: 600; font-size: .88rem; cursor: pointer; border: 0;
background: transparent; color: var(--muted); padding: 8px 14px; border-radius: 9px;
transition: background .2s, color .2s;
}
.tab:hover { color: var(--ink); }
.tab.is-active { background: var(--surface); color: var(--ink); box-shadow: var(--shadow-sm); }
.tabpanels { margin-top: 16px; }
.tabpanel { color: var(--muted); font-size: 1rem; }
.tabpanel[hidden] { display: none; }
/* Compare band */
.compare {
background: var(--surface); border: 1px solid var(--line); border-radius: 22px;
padding: 44px clamp(20px, 5vw, 48px); box-shadow: var(--shadow); text-align: center;
scroll-margin-top: 84px;
}
.compare h2 { font-size: clamp(1.5rem, 3.4vw, 2.1rem); font-weight: 800; }
.compare-sub { color: var(--muted); margin: 10px 0 30px; }
.compare-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 18px; text-align: left; }
.compare-col { border: 1px solid var(--line); border-radius: 16px; padding: 22px; }
.compare-col.old { background: var(--surface-2); }
.compare-col.now { background: color-mix(in srgb, var(--brand) 7%, var(--surface)); border-color: color-mix(in srgb, var(--brand) 28%, var(--line)); }
.cc-head { display: flex; align-items: center; gap: 8px; font-weight: 800; font-size: 1.05rem; margin: 0 0 14px; }
.compare-col ul { list-style: none; margin: 0; padding: 0; display: grid; gap: 11px; }
.compare-col li { position: relative; padding-left: 26px; font-size: .96rem; }
.compare-col.old li::before { content: "✕"; position: absolute; left: 0; color: var(--danger); font-weight: 800; }
.compare-col.now li::before { content: "✓"; position: absolute; left: 0; color: var(--ok); font-weight: 800; }
/* CTA */
.cta { padding: 80px 0; text-align: center; }
.cta-inner { max-width: 620px; }
.cta h2 { font-size: clamp(1.7rem, 4vw, 2.6rem); font-weight: 800; }
.cta > .cta-inner > p { color: var(--muted); margin-top: 12px; }
.cta-form { display: flex; gap: 10px; justify-content: center; margin-top: 26px; flex-wrap: wrap; }
.cta-form input {
font: inherit; padding: 13px 16px; border-radius: 14px; border: 1px solid var(--line);
background: var(--surface); color: var(--ink); min-width: 260px; flex: 1 1 260px; max-width: 360px;
}
.cta-form input:focus-visible { border-color: var(--brand); }
.cta-form input.invalid { border-color: var(--danger); }
.cta-note { min-height: 1.4em; margin-top: 14px; font-weight: 600; font-size: .92rem; color: var(--ok); }
.cta-note.err { color: var(--danger); }
/* Footer */
.foot { border-top: 1px solid var(--line); padding: 26px 0; color: var(--muted); font-size: .88rem; }
.foot-inner { display: flex; justify-content: space-between; gap: 14px; flex-wrap: wrap; }
.foot-links a { text-decoration: none; font-weight: 600; }
.foot-links a:hover { color: var(--ink); }
/* Toast */
.toast {
position: fixed; left: 50%; bottom: 24px; transform: translate(-50%, 24px);
background: var(--ink); color: var(--bg); padding: 12px 18px; border-radius: 12px;
font-weight: 600; font-size: .92rem; box-shadow: var(--shadow); opacity: 0;
pointer-events: none; transition: opacity .25s, transform .25s; z-index: 80; max-width: 90vw;
}
.toast.show { opacity: 1; transform: translate(-50%, 0); }
/* Responsive */
@media (max-width: 920px) {
.tour-layout { grid-template-columns: 1fr; }
.spy { display: none; }
.feature, .feature-stack { grid-template-columns: 1fr; gap: 28px; }
.feature-rev .feature-copy { order: 0; }
.features { gap: 64px; }
}
@media (max-width: 680px) {
.topnav { display: none; }
#themeToggle .theme-label { display: none; }
.compare-grid { grid-template-columns: 1fr; }
}
@media (max-width: 400px) {
.board-cols { grid-template-columns: 1fr; }
.hero { padding-top: 44px; }
}(function () {
"use strict";
/* ---------- Toast helper ---------- */
var toastEl = document.getElementById("toast");
var toastTimer;
function toast(msg) {
if (!toastEl) return;
toastEl.textContent = msg;
toastEl.classList.add("show");
clearTimeout(toastTimer);
toastTimer = setTimeout(function () { toastEl.classList.remove("show"); }, 2600);
}
/* ---------- Theme toggle ---------- */
var root = document.documentElement;
var themeBtn = document.getElementById("themeToggle");
function applyTheme(theme) {
root.setAttribute("data-theme", theme);
var dark = theme === "dark";
themeBtn.setAttribute("aria-pressed", String(dark));
themeBtn.querySelector(".theme-ico").textContent = dark ? "☀️" : "🌙";
themeBtn.querySelector(".theme-label").textContent = dark ? "Light" : "Dark";
}
var stored = null;
try { stored = localStorage.getItem("cadence-theme"); } catch (e) {}
applyTheme(stored || (window.matchMedia && window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light"));
themeBtn.addEventListener("click", function () {
var next = root.getAttribute("data-theme") === "dark" ? "light" : "dark";
applyTheme(next);
try { localStorage.setItem("cadence-theme", next); } catch (e) {}
});
/* ---------- Smooth-scroll for in-page anchors ---------- */
document.querySelectorAll('a[href^="#"]').forEach(function (a) {
a.addEventListener("click", function (e) {
var id = a.getAttribute("href");
if (id.length < 2) return;
var target = document.querySelector(id);
if (!target) return;
e.preventDefault();
target.scrollIntoView({ behavior: "smooth", block: "start" });
history.replaceState(null, "", id);
});
});
/* ---------- Scrollspy ---------- */
var spyLinks = Array.prototype.slice.call(document.querySelectorAll(".spy-list a[data-spy]"));
var spyTargets = spyLinks
.map(function (l) { return document.getElementById(l.getAttribute("data-spy")); })
.filter(Boolean);
function setActiveSpy(id) {
spyLinks.forEach(function (l) {
l.classList.toggle("is-active", l.getAttribute("data-spy") === id);
});
}
if ("IntersectionObserver" in window && spyTargets.length) {
var visible = {};
var io = new IntersectionObserver(function (entries) {
entries.forEach(function (en) { visible[en.target.id] = en.isIntersecting ? en.intersectionRatio : 0; });
// pick the most-visible section
var best = null, bestRatio = 0;
spyTargets.forEach(function (t) {
var r = visible[t.id] || 0;
if (r > bestRatio) { bestRatio = r; best = t.id; }
});
if (best) setActiveSpy(best);
}, { rootMargin: "-45% 0px -45% 0px", threshold: [0, 0.25, 0.5, 1] });
spyTargets.forEach(function (t) { io.observe(t); });
}
/* ---------- Automation tabs (swap mock) ---------- */
var FLOWS = {
route: {
title: "Rule · Smart routing",
nodes: [
{ cls: "fn-trigger", ico: "📥", t: "When an issue is created", s: "Trigger" },
{ cls: "fn-cond", ico: "🔀", t: "If label is “billing”", s: "Condition" },
{ cls: "fn-action", ico: "🎯", t: "Assign to Payments squad", s: "Action" }
]
},
notify: {
title: "Rule · Notifications",
nodes: [
{ cls: "fn-trigger", ico: "🔔", t: "When status → In review", s: "Trigger" },
{ cls: "fn-cond", ico: "🕘", t: "Unless it’s quiet hours", s: "Condition" },
{ cls: "fn-action", ico: "💬", t: "Post to #eng-reviews", s: "Action" }
]
},
sla: {
title: "Rule · SLA guardrails",
nodes: [
{ cls: "fn-trigger", ico: "⏱️", t: "When due in < 4 hours", s: "Trigger" },
{ cls: "fn-cond", ico: "🚧", t: "If still unassigned", s: "Condition" },
{ cls: "fn-action", ico: "🚨", t: "Escalate + log breach", s: "Action" }
]
}
};
var flowEl = document.getElementById("flow");
var flowTitle = document.getElementById("flowTitle");
function renderFlow(key) {
var spec = FLOWS[key] || FLOWS.route;
if (flowTitle) flowTitle.textContent = spec.title;
var html = "";
spec.nodes.forEach(function (n, i) {
html +=
'<div class="flow-node ' + n.cls + '">' +
'<span class="fn-ico" aria-hidden="true">' + n.ico + "</span>" +
"<span>" + n.t + "<small>" + n.s + "</small></span>" +
"</div>";
if (i < spec.nodes.length - 1) html += '<div class="flow-arrow" aria-hidden="true">↓</div>';
});
flowEl.innerHTML = html;
}
renderFlow("route");
var tabs = Array.prototype.slice.call(document.querySelectorAll(".tab"));
function selectTab(tab) {
tabs.forEach(function (t) {
var on = t === tab;
t.classList.toggle("is-active", on);
t.setAttribute("aria-selected", String(on));
t.tabIndex = on ? 0 : -1;
var panel = document.getElementById(t.getAttribute("aria-controls"));
if (panel) {
panel.hidden = !on;
panel.classList.toggle("is-active", on);
}
});
renderFlow(tab.getAttribute("data-mock"));
}
tabs.forEach(function (tab, idx) {
tab.addEventListener("click", function () { selectTab(tab); });
tab.addEventListener("keydown", function (e) {
var dir = e.key === "ArrowRight" ? 1 : e.key === "ArrowLeft" ? -1 : 0;
if (!dir) return;
e.preventDefault();
var next = tabs[(idx + dir + tabs.length) % tabs.length];
selectTab(next);
next.focus();
});
});
/* ---------- CTA form ---------- */
var form = document.getElementById("ctaForm");
var input = document.getElementById("email");
var note = document.getElementById("ctaNote");
form.addEventListener("submit", function (e) {
e.preventDefault();
var val = input.value.trim();
var ok = /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(val);
if (!ok) {
input.classList.add("invalid");
note.classList.add("err");
note.textContent = "Please enter a valid work email.";
input.focus();
return;
}
input.classList.remove("invalid");
note.classList.remove("err");
note.textContent = "Trial ready — check " + val + " to confirm.";
toast("Welcome aboard! 14-day trial started.");
form.reset();
});
input.addEventListener("input", function () {
if (input.classList.contains("invalid")) {
input.classList.remove("invalid");
note.textContent = "";
note.classList.remove("err");
}
});
})();<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>Cadence — Product Tour</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=Inter:wght@400;500;600;700;800&display=swap" rel="stylesheet" />
<link rel="stylesheet" href="style.css" />
</head>
<body>
<a class="skip-link" href="#main">Skip to content</a>
<header class="topbar" role="banner">
<div class="wrap topbar-inner">
<a class="brand" href="#top" aria-label="Cadence home">
<span class="brand-mark" aria-hidden="true">
<svg viewBox="0 0 24 24" width="22" height="22"><path d="M4 14c3-7 5-7 8 0s5 7 8 0" fill="none" stroke="currentColor" stroke-width="2.4" stroke-linecap="round"/></svg>
</span>
<span class="brand-name">Cadence</span>
</a>
<nav class="topnav" aria-label="Primary">
<a href="#workflows">Workflows</a>
<a href="#insights">Insights</a>
<a href="#automation">Automation</a>
<a href="#compare">Why Cadence</a>
</nav>
<div class="topbar-actions">
<button class="btn btn-ghost" id="themeToggle" type="button" aria-pressed="false">
<span class="theme-ico" aria-hidden="true">🌙</span><span class="theme-label">Dark</span>
</button>
<a class="btn btn-primary" href="#cta">Start free</a>
</div>
</div>
</header>
<main id="main">
<section class="hero" id="top">
<div class="wrap hero-inner">
<p class="eyebrow">Product tour</p>
<h1>See exactly how teams move faster with Cadence.</h1>
<p class="lede">A single workspace for planning, tracking, and automating the work that keeps your product shipping — with insights that actually tell you what to do next.</p>
<div class="hero-cta">
<a class="btn btn-primary btn-lg" href="#cta">Start free trial</a>
<a class="btn btn-ghost btn-lg" href="#workflows">Take the 2-min tour</a>
</div>
<ul class="trust-row" aria-label="Trusted by">
<li>4,200+ teams</li>
<li aria-hidden="true">·</li>
<li>SOC 2 Type II</li>
<li aria-hidden="true">·</li>
<li>99.98% uptime</li>
</ul>
</div>
</section>
<div class="tour-layout wrap">
<nav class="spy" aria-label="Feature sections">
<p class="spy-title">On this page</p>
<ol class="spy-list">
<li><a href="#workflows" data-spy="workflows" class="is-active">Visual workflows</a></li>
<li><a href="#insights" data-spy="insights">Live insights</a></li>
<li><a href="#automation" data-spy="automation">Automation</a></li>
<li><a href="#compare" data-spy="compare">Why Cadence</a></li>
</ol>
</nav>
<div class="features">
<!-- FEATURE 1 -->
<section class="feature" id="workflows" aria-labelledby="f1-title">
<div class="feature-copy">
<span class="kicker kicker-1">Plan</span>
<h2 id="f1-title">Workflows you can actually see</h2>
<p>Drag work across stages, group by owner or sprint, and watch dependencies update in real time. No more guessing what's blocked.</p>
<ul class="bullets">
<li>Board, timeline, and table views of the same data</li>
<li>Automatic WIP limits keep columns honest</li>
<li>Dependency lines flag risk before it lands</li>
</ul>
<a class="link-arrow" href="#cta">Explore the board <span aria-hidden="true">→</span></a>
</div>
<div class="feature-mock">
<div class="mock mock-board" role="img" aria-label="Kanban board with three columns of task cards">
<div class="mock-head"><span class="dot r"></span><span class="dot y"></span><span class="dot g"></span><span class="mock-title">Sprint 24 · Board</span></div>
<div class="board-cols">
<div class="bcol"><p class="bcol-h">To do <span>4</span></p>
<div class="card"><span class="tag tag-blue">Design</span><p>Onboarding empty states</p><div class="avatar a1"></div></div>
<div class="card"><span class="tag tag-violet">Eng</span><p>Webhook retries</p><div class="avatar a2"></div></div>
</div>
<div class="bcol"><p class="bcol-h">Doing <span>2</span></p>
<div class="card hot"><span class="tag tag-amber">Eng</span><p>Realtime presence</p><div class="bar"><i style="width:64%"></i></div><div class="avatar a3"></div></div>
</div>
<div class="bcol"><p class="bcol-h">Done <span>9</span></p>
<div class="card done"><span class="tag tag-green">QA</span><p>SSO hardening</p><div class="avatar a1"></div></div>
</div>
</div>
</div>
</div>
</section>
<!-- FEATURE 2 -->
<section class="feature feature-rev" id="insights" aria-labelledby="f2-title">
<div class="feature-copy">
<span class="kicker kicker-2">Measure</span>
<h2 id="f2-title">Insights that point at the next move</h2>
<p>Velocity, cycle time, and forecast in one glance — recalculated as work moves, so reviews start with answers, not spreadsheets.</p>
<ul class="bullets">
<li>Cycle-time trend with anomaly highlights</li>
<li>Probabilistic ship-date forecasting</li>
<li>Shareable read-only dashboards</li>
</ul>
<a class="link-arrow" href="#cta">See the dashboard <span aria-hidden="true">→</span></a>
</div>
<div class="feature-mock">
<div class="mock mock-chart" role="img" aria-label="Analytics card showing a rising velocity line chart and key metrics">
<div class="mock-head"><span class="dot r"></span><span class="dot y"></span><span class="dot g"></span><span class="mock-title">Insights · Last 8 weeks</span></div>
<div class="chart-stats">
<div><p class="cs-k">Velocity</p><p class="cs-v">38 <em class="up">+12%</em></p></div>
<div><p class="cs-k">Cycle time</p><p class="cs-v">2.4d <em class="down">-0.6d</em></p></div>
<div><p class="cs-k">Forecast</p><p class="cs-v">Jun 28</p></div>
</div>
<svg class="spark" viewBox="0 0 320 120" preserveAspectRatio="none" aria-hidden="true">
<defs><linearGradient id="g" x1="0" y1="0" x2="0" y2="1">
<stop offset="0" stop-color="var(--brand)" stop-opacity=".28"/>
<stop offset="1" stop-color="var(--brand)" stop-opacity="0"/>
</linearGradient></defs>
<path d="M0 96 L46 84 L92 90 L138 64 L184 70 L230 40 L276 48 L320 18 L320 120 L0 120 Z" fill="url(#g)"/>
<path d="M0 96 L46 84 L92 90 L138 64 L184 70 L230 40 L276 48 L320 18" fill="none" stroke="var(--brand)" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="320" cy="18" r="4" fill="var(--brand)"/>
</svg>
</div>
</div>
</section>
<!-- FEATURE 3 — interactive tabs swap mock -->
<section class="feature feature-stack" id="automation" aria-labelledby="f3-title">
<div class="feature-copy">
<span class="kicker kicker-3">Automate</span>
<h2 id="f3-title">Automation for the busywork</h2>
<p>Build no-code rules that route, notify, and update. Pick a capability to preview how it works.</p>
<div class="tabs" role="tablist" aria-label="Automation capabilities">
<button class="tab is-active" role="tab" id="tab-route" aria-selected="true" aria-controls="panel-route" data-mock="route">Smart routing</button>
<button class="tab" role="tab" id="tab-notify" aria-selected="false" aria-controls="panel-notify" tabindex="-1" data-mock="notify">Notifications</button>
<button class="tab" role="tab" id="tab-sla" aria-selected="false" aria-controls="panel-sla" tabindex="-1" data-mock="sla">SLA guardrails</button>
</div>
<div class="tabpanels">
<div class="tabpanel is-active" role="tabpanel" id="panel-route" aria-labelledby="tab-route">Auto-assign incoming work to the right squad by label, repo, or customer tier.</div>
<div class="tabpanel" role="tabpanel" id="panel-notify" aria-labelledby="tab-notify" hidden>Ping the right channel when status flips, with quiet hours respected.</div>
<div class="tabpanel" role="tabpanel" id="panel-sla" aria-labelledby="tab-sla" hidden>Escalate before a deadline slips and log the breach automatically.</div>
</div>
</div>
<div class="feature-mock">
<div class="mock mock-flow" role="img" aria-label="Automation rule builder preview">
<div class="mock-head"><span class="dot r"></span><span class="dot y"></span><span class="dot g"></span><span class="mock-title" id="flowTitle">Rule · Smart routing</span></div>
<div class="flow" id="flow"><!-- injected by JS --></div>
</div>
</div>
</section>
<!-- COMPARE BAND -->
<section class="compare" id="compare" aria-labelledby="cmp-title">
<h2 id="cmp-title">The old way vs. Cadence</h2>
<p class="compare-sub">Same team, half the overhead.</p>
<div class="compare-grid">
<div class="compare-col old">
<p class="cc-head">Before <span aria-hidden="true">😮💨</span></p>
<ul>
<li>Status updates scattered across chats</li>
<li>Spreadsheets that go stale by Monday</li>
<li>Forecasts that are really just hope</li>
<li>Manual hand-offs that fall through</li>
</ul>
</div>
<div class="compare-col now">
<p class="cc-head">With Cadence <span aria-hidden="true">⚡</span></p>
<ul>
<li>One source of truth, always current</li>
<li>Metrics recalculated as work moves</li>
<li>Probabilistic, defensible forecasts</li>
<li>Automated routing and escalation</li>
</ul>
</div>
</div>
</section>
</div>
</div>
<section class="cta" id="cta" aria-labelledby="cta-title">
<div class="wrap cta-inner">
<h2 id="cta-title">Ship with confidence — start in minutes.</h2>
<p>Free for 14 days. No card required. Cancel anytime.</p>
<form class="cta-form" id="ctaForm" novalidate>
<label class="sr-only" for="email">Work email</label>
<input id="email" name="email" type="email" placeholder="[email protected]" autocomplete="email" required />
<button class="btn btn-primary btn-lg" type="submit">Start free trial</button>
</form>
<p class="cta-note" id="ctaNote" role="status"></p>
</div>
</section>
</main>
<footer class="foot" role="contentinfo">
<div class="wrap foot-inner">
<span>© 2026 Cadence Labs — fictional demo</span>
<span class="foot-links"><a href="#top">Back to top</a></span>
</div>
</footer>
<div class="toast" id="toast" role="status" aria-live="polite"></div>
<script src="script.js"></script>
</body>
</html>Features / Product Tour
A polished product-tour page for a fictional planning tool, “Cadence.” Alternating feature sections pair benefit-led copy with inline-SVG mockups — a kanban board, a velocity insights card with a CSS/SVG sparkline, and an automation rule builder. A sticky sub-nav on the left scrollspies the feature list, highlighting whichever section is in view, and every nav link smooth-scrolls to its target.
The automation section uses an accessible tablist: clicking a tab (or arrow-keying through it) swaps the rule-builder preview to show smart routing, notifications, or SLA guardrails. A before-and-after comparison band contrasts the “old way” with Cadence, and the closing CTA validates the email field inline and confirms with a toast.
A header theme toggle flips the whole page between light and dark with the preference persisted, and the layout collapses gracefully to a single column on small screens.
Illustrative SaaS UI only — fictional product, metrics, and billing. No real backend.