Coworking — Creative Loft Landing
A warm industrial landing page for Foundry Loft, a fictional creative coworking studio in a converted warehouse. A raw hero pairs concrete-grid texture and amber afternoon light with count-up stats and loft photo blocks, then sections walk through who it serves, the shared studio gear with live availability chips, a community-of-makers feed, monthly or annual pricing, and a validated book-a-tour form. Built in vanilla HTML, CSS and JavaScript with scroll reveal, a mobile menu and toast feedback.
MCP
Code
:root {
--concrete: #efeae3;
--concrete-d: #e2dcd2;
--amber: #e8902b;
--amber-d: #cc7918;
--amber-50: #fdf1e2;
--char: #1c1b19;
--ink: #26241f;
--ink-2: #4a463e;
--muted: #7b766c;
--bg: #f6f3ee;
--surface: #ffffff;
--plant: #5f7a52;
--line: rgba(28, 27, 25, 0.1);
--line-2: rgba(28, 27, 25, 0.18);
--ok: #2f9e6f;
--warn: #d98a2b;
--danger: #d4503e;
--r-sm: 8px;
--r-md: 14px;
--r-lg: 20px;
--shadow: 0 1px 2px rgba(28, 27, 25, 0.06), 0 8px 24px rgba(28, 27, 25, 0.07);
--shadow-lg: 0 20px 50px rgba(28, 27, 25, 0.16);
}
* { box-sizing: border-box; margin: 0; padding: 0; }
html { scroll-behavior: smooth; scroll-padding-top: 78px; }
body {
font-family: "Inter", system-ui, -apple-system, sans-serif;
background: var(--bg);
color: var(--ink);
line-height: 1.5;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
overflow-x: hidden;
}
.wrap { width: 100%; max-width: 1180px; margin: 0 auto; padding: 0 24px; }
img { max-width: 100%; display: block; }
a { color: inherit; text-decoration: none; }
::selection { background: var(--amber); color: #fff; }
/* ---------- buttons ---------- */
.btn {
--b: var(--char);
display: inline-flex; align-items: center; justify-content: center; gap: 8px;
font: inherit; font-weight: 600; font-size: 0.95rem;
padding: 11px 20px; border-radius: var(--r-sm);
border: 1.5px solid transparent; cursor: pointer;
transition: transform 0.16s ease, background 0.16s ease, box-shadow 0.16s ease, color 0.16s ease;
white-space: nowrap; letter-spacing: 0.01em;
}
.btn:active { transform: translateY(1px) scale(0.99); }
.btn:focus-visible { outline: 3px solid var(--amber); outline-offset: 2px; }
.btn--lg { padding: 14px 26px; font-size: 1rem; }
.btn--block { width: 100%; }
.btn--amber { background: var(--amber); color: #fff; box-shadow: 0 6px 18px rgba(232, 144, 43, 0.32); }
.btn--amber:hover { background: var(--amber-d); transform: translateY(-2px); }
.btn--line { background: transparent; border-color: var(--line-2); color: var(--ink); }
.btn--line:hover { border-color: var(--char); background: var(--char); color: #fff; transform: translateY(-2px); }
.btn--ghost { background: transparent; color: var(--ink); border-color: var(--line-2); }
.btn--ghost:hover { background: var(--char); color: #fff; border-color: var(--char); }
.kicker {
display: inline-block; font-size: 0.74rem; font-weight: 700; letter-spacing: 0.18em;
text-transform: uppercase; color: var(--ink-2);
padding: 5px 11px; border: 1px solid var(--line-2); border-radius: 100px; margin-bottom: 16px;
}
.kicker--amber { color: var(--amber-d); border-color: rgba(232, 144, 43, 0.4); background: var(--amber-50); }
.eyebrow {
display: inline-block; font-size: 0.78rem; font-weight: 600; letter-spacing: 0.04em;
color: var(--amber-d); background: var(--amber-50);
padding: 6px 13px; border-radius: 100px; margin-bottom: 22px;
border: 1px solid rgba(232, 144, 43, 0.3);
}
/* ---------- nav ---------- */
.nav {
position: sticky; top: 0; z-index: 50;
background: rgba(246, 243, 238, 0.82);
backdrop-filter: saturate(1.4) blur(12px);
border-bottom: 1px solid transparent;
transition: border-color 0.25s ease, box-shadow 0.25s ease, background 0.25s ease;
}
.nav.is-scrolled { border-bottom-color: var(--line); box-shadow: 0 4px 20px rgba(28, 27, 25, 0.05); }
.nav__inner { display: flex; align-items: center; gap: 20px; height: 66px; }
.brand { display: inline-flex; align-items: center; gap: 10px; font-weight: 800; }
.brand__mark { color: var(--amber); font-size: 1.05rem; letter-spacing: -2px; }
.brand__name { font-size: 1.02rem; letter-spacing: 0.04em; color: var(--char); }
.brand__sub { color: var(--amber-d); }
.nav__links { display: flex; gap: 28px; margin-left: auto; }
.nav__links a {
font-size: 0.92rem; font-weight: 500; color: var(--ink-2); position: relative; padding: 4px 0;
}
.nav__links a::after {
content: ""; position: absolute; left: 0; bottom: -2px; width: 0; height: 2px;
background: var(--amber); transition: width 0.2s ease;
}
.nav__links a:hover { color: var(--char); }
.nav__links a:hover::after { width: 100%; }
.nav__cta { display: flex; }
.nav__burger {
display: none; flex-direction: column; gap: 5px; background: none; border: 0;
cursor: pointer; padding: 8px; margin-left: auto;
}
.nav__burger span { width: 24px; height: 2px; background: var(--char); border-radius: 2px; transition: transform 0.25s ease, opacity 0.2s ease; }
.nav__burger.is-open span:nth-child(1) { transform: translateY(7px) rotate(45deg); }
.nav__burger.is-open span:nth-child(2) { opacity: 0; }
.nav__burger.is-open span:nth-child(3) { transform: translateY(-7px) rotate(-45deg); }
.mobile-nav {
display: none; flex-direction: column; gap: 4px; padding: 12px 24px 22px;
border-top: 1px solid var(--line); background: var(--bg);
}
.mobile-nav a { padding: 12px 4px; font-weight: 600; color: var(--ink); border-bottom: 1px solid var(--line); }
.mobile-nav a:last-child { border: 0; margin-top: 8px; }
.mobile-nav.is-open { display: flex; }
/* ---------- hero ---------- */
.hero { padding: 64px 0 56px; position: relative; }
.hero::before {
content: ""; position: absolute; inset: 0; z-index: -1; opacity: 0.5;
background-image: linear-gradient(var(--line) 1px, transparent 1px), linear-gradient(90deg, var(--line) 1px, transparent 1px);
background-size: 38px 38px;
-webkit-mask-image: radial-gradient(ellipse 80% 60% at 30% 40%, #000, transparent 75%);
mask-image: radial-gradient(ellipse 80% 60% at 30% 40%, #000, transparent 75%);
}
.hero__grid { display: grid; grid-template-columns: 1.05fr 0.95fr; gap: 56px; align-items: center; }
.hero__copy h1 {
font-size: clamp(2.3rem, 5.4vw, 3.7rem); line-height: 1.04; font-weight: 800;
letter-spacing: -0.02em; color: var(--char); margin-bottom: 20px;
}
.hero__copy h1 em { font-style: normal; color: var(--amber-d); position: relative; }
.hero__copy h1 em::after {
content: ""; position: absolute; left: 0; right: 0; bottom: 2px; height: 10px;
background: rgba(232, 144, 43, 0.22); z-index: -1; border-radius: 2px;
}
.hero__lede { font-size: 1.1rem; color: var(--ink-2); max-width: 36ch; margin-bottom: 28px; }
.hero__actions { display: flex; gap: 12px; flex-wrap: wrap; margin-bottom: 38px; }
.hero__stats { display: flex; gap: 30px; flex-wrap: wrap; padding-top: 26px; border-top: 1px solid var(--line); }
.hero__stats dt { font-size: 1.9rem; font-weight: 800; color: var(--char); letter-spacing: -0.02em; line-height: 1; }
.hero__stats dd { font-size: 0.82rem; color: var(--muted); margin-top: 6px; font-weight: 500; }
.hero__art { position: relative; display: grid; grid-template-columns: 1fr 1fr; grid-template-rows: auto auto; gap: 14px; }
.photo {
border-radius: var(--r-md); position: relative; overflow: hidden;
box-shadow: var(--shadow); border: 1px solid var(--line);
transition: transform 0.3s ease;
}
.photo::after {
content: attr(data-label); position: absolute; left: 12px; bottom: 12px;
font-size: 0.72rem; font-weight: 600; color: #fff; letter-spacing: 0.02em;
background: rgba(28, 27, 25, 0.55); padding: 4px 9px; border-radius: 100px;
backdrop-filter: blur(4px);
}
.hero__art:hover .photo { transform: scale(0.985); }
.photo:hover { transform: scale(1.02) !important; z-index: 2; }
.photo--tall { grid-row: 1 / 3; min-height: 340px; background: linear-gradient(150deg, #d8cfc1 0%, #c4b6a2 55%, #8f8170 100%); }
.photo--amber { min-height: 158px; background: linear-gradient(140deg, var(--amber) 0%, var(--amber-d) 60%, #9c5d12 100%); }
.photo--char { min-height: 168px; background: linear-gradient(140deg, #3a3833 0%, var(--char) 100%); }
.art__tag {
position: absolute; top: -14px; right: -10px; z-index: 3; transform: rotate(6deg);
background: var(--char); color: #fff; font-size: 0.74rem; font-weight: 700;
letter-spacing: 0.06em; padding: 7px 13px; border-radius: 100px; box-shadow: var(--shadow);
}
/* ---------- marquee ---------- */
.marquee { overflow: hidden; background: var(--char); color: var(--concrete); padding: 14px 0; border-block: 1px solid var(--char); }
.marquee__track { display: flex; gap: 26px; white-space: nowrap; width: max-content; animation: scroll 26s linear infinite; }
.marquee__track span { font-size: 0.92rem; font-weight: 700; letter-spacing: 0.14em; }
.marquee__track span:nth-child(even) { color: var(--amber); }
.marquee:hover .marquee__track { animation-play-state: paused; }
@keyframes scroll { to { transform: translateX(-50%); } }
/* ---------- sections ---------- */
.section { padding: 84px 0; }
.section--char { background: var(--char); color: var(--concrete); }
.section--concrete { background: var(--concrete); }
.sec-head { max-width: 620px; margin-bottom: 44px; }
.sec-head h2 { font-size: clamp(1.7rem, 3.6vw, 2.4rem); font-weight: 800; letter-spacing: -0.02em; line-height: 1.12; color: var(--char); margin-bottom: 14px; }
.sec-head p { color: var(--ink-2); font-size: 1.04rem; }
.sec-head--light h2 { color: var(--concrete); }
.sec-head--light p { color: rgba(239, 234, 227, 0.7); }
/* ---------- who-grid ---------- */
.who-grid { display: grid; grid-template-columns: repeat(4, 1fr); gap: 18px; }
.who-card {
background: var(--surface); border: 1px solid var(--line); border-radius: var(--r-md);
padding: 26px 22px; transition: transform 0.2s ease, box-shadow 0.2s ease, border-color 0.2s ease;
position: relative; overflow: hidden; cursor: default;
}
.who-card::before {
content: ""; position: absolute; top: 0; left: 0; width: 100%; height: 3px;
background: var(--amber); transform: scaleX(0); transform-origin: left; transition: transform 0.25s ease;
}
.who-card:hover, .who-card:focus-visible { transform: translateY(-6px); box-shadow: var(--shadow-lg); border-color: transparent; outline: none; }
.who-card:hover::before, .who-card:focus-visible::before { transform: scaleX(1); }
.who-card__icon {
display: inline-grid; place-items: center; width: 46px; height: 46px;
background: var(--amber-50); color: var(--amber-d); border-radius: var(--r-sm);
font-size: 1.35rem; margin-bottom: 16px;
}
.who-card h3 { font-size: 1.08rem; font-weight: 700; color: var(--char); margin-bottom: 8px; }
.who-card p { font-size: 0.92rem; color: var(--ink-2); margin-bottom: 16px; }
.who-card__tag { font-size: 0.74rem; font-weight: 700; color: var(--plant); letter-spacing: 0.03em; }
/* ---------- amenities ---------- */
.amen-grid { display: grid; grid-template-columns: repeat(3, 1fr); gap: 16px; }
.amen-card {
background: rgba(255, 255, 255, 0.04); border: 1px solid rgba(239, 234, 227, 0.14);
border-radius: var(--r-md); padding: 22px; cursor: pointer; text-align: left;
color: var(--concrete); font: inherit; transition: background 0.2s ease, border-color 0.2s ease, transform 0.2s ease;
display: flex; flex-direction: column; gap: 10px;
}
.amen-card:hover { background: rgba(239, 234, 227, 0.07); border-color: rgba(232, 144, 43, 0.5); transform: translateY(-4px); }
.amen-card:focus-visible { outline: 3px solid var(--amber); outline-offset: 2px; }
.amen-card__head { display: flex; align-items: center; justify-content: space-between; gap: 10px; }
.amen-card__icon { font-size: 1.5rem; }
.amen-card h3 { font-size: 1.04rem; font-weight: 700; }
.amen-card p { font-size: 0.88rem; color: rgba(239, 234, 227, 0.62); }
.amen-status {
display: inline-flex; align-items: center; gap: 6px; font-size: 0.74rem; font-weight: 700;
padding: 4px 10px; border-radius: 100px; letter-spacing: 0.02em; align-self: flex-start;
}
.amen-status::before { content: ""; width: 7px; height: 7px; border-radius: 50%; background: currentColor; box-shadow: 0 0 0 3px rgba(255,255,255,0.12); }
.amen-status.is-free { color: #6fe0a8; background: rgba(47, 158, 111, 0.16); }
.amen-status.is-busy { color: #f0917f; background: rgba(212, 80, 62, 0.16); }
.amen-status.is-soon { color: #f0b66a; background: rgba(217, 138, 43, 0.16); }
.amen-note { margin-top: 22px; font-size: 0.86rem; color: rgba(239, 234, 227, 0.55); }
/* ---------- makers ---------- */
.makers__grid { display: grid; grid-template-columns: 1fr 0.85fr; gap: 54px; align-items: center; }
.makers__copy h2 { font-size: clamp(1.7rem, 3.6vw, 2.3rem); font-weight: 800; letter-spacing: -0.02em; color: var(--char); margin-bottom: 16px; line-height: 1.14; }
.makers__copy > p { color: var(--ink-2); font-size: 1.04rem; margin-bottom: 24px; }
.makers__list { list-style: none; display: grid; gap: 12px; margin-bottom: 28px; }
.makers__list li { font-size: 0.96rem; color: var(--ink-2); padding-left: 26px; position: relative; }
.makers__list li::before { content: "◆"; position: absolute; left: 0; color: var(--amber); font-size: 0.7rem; top: 4px; }
.makers__list strong { color: var(--char); }
.avi {
display: inline-grid; place-items: center; width: 44px; height: 44px; border-radius: 50%;
font-size: 0.82rem; font-weight: 700; color: #fff; border: 2.5px solid var(--bg); flex: none;
}
.makers__avatars { display: flex; }
.makers__avatars .avi { margin-left: -10px; box-shadow: 0 2px 6px rgba(0,0,0,0.12); }
.makers__avatars .avi:first-child { margin-left: 0; }
.avi--a { background: linear-gradient(135deg, #e8902b, #cc7918); }
.avi--b { background: linear-gradient(135deg, #5f7a52, #455c3b); }
.avi--c { background: linear-gradient(135deg, #d4503e, #a83a2c); }
.avi--d { background: linear-gradient(135deg, #4a463e, #1c1b19); }
.avi--e { background: linear-gradient(135deg, #7b766c, #4a463e); }
.avi--more { background: var(--concrete-d); color: var(--ink-2); font-size: 0.78rem; }
.makers__feed { display: grid; gap: 14px; }
.feed-card { background: var(--surface); border: 1px solid var(--line); border-radius: var(--r-md); padding: 18px; box-shadow: var(--shadow); transition: transform 0.2s ease; }
.feed-card:hover { transform: translateX(4px); }
.feed-card__top { display: flex; align-items: center; gap: 12px; margin-bottom: 12px; }
.feed-card__top .avi { width: 38px; height: 38px; border: 0; }
.feed-card__top strong { display: block; font-size: 0.92rem; color: var(--char); }
.feed-card__top span { font-size: 0.78rem; color: var(--muted); }
.feed-card p { font-size: 0.92rem; color: var(--ink-2); margin-bottom: 12px; }
.feed-card__meta { font-size: 0.78rem; color: var(--muted); font-weight: 600; }
/* ---------- pricing ---------- */
.bill-toggle { display: inline-flex; gap: 4px; background: var(--surface); border: 1px solid var(--line-2); border-radius: 100px; padding: 4px; margin-bottom: 36px; }
.bill-toggle__btn { font: inherit; font-weight: 600; font-size: 0.9rem; padding: 8px 20px; border: 0; border-radius: 100px; background: transparent; color: var(--ink-2); cursor: pointer; transition: background 0.2s ease, color 0.2s ease; }
.bill-toggle__btn span { color: var(--plant); font-weight: 700; font-size: 0.8rem; }
.bill-toggle__btn.is-active { background: var(--char); color: #fff; }
.bill-toggle__btn.is-active span { color: #9bd1a8; }
.price-grid { display: grid; grid-template-columns: repeat(3, 1fr); gap: 18px; align-items: start; }
.price-card { background: var(--surface); border: 1px solid var(--line); border-radius: var(--r-lg); padding: 30px 26px; position: relative; transition: transform 0.2s ease, box-shadow 0.2s ease; }
.price-card:hover { transform: translateY(-5px); box-shadow: var(--shadow-lg); }
.price-card--feature { border: 2px solid var(--char); background: var(--char); color: var(--concrete); transform: scale(1.03); }
.price-card--feature:hover { transform: scale(1.03) translateY(-5px); }
.price-card__flag { position: absolute; top: -13px; left: 50%; transform: translateX(-50%); background: var(--amber); color: #fff; font-size: 0.72rem; font-weight: 700; letter-spacing: 0.04em; padding: 5px 14px; border-radius: 100px; white-space: nowrap; box-shadow: 0 6px 16px rgba(232,144,43,0.4); }
.price-card h3 { font-size: 1.2rem; font-weight: 800; margin-bottom: 6px; color: var(--char); }
.price-card--feature h3 { color: var(--concrete); }
.price-card__desc { font-size: 0.9rem; color: var(--muted); margin-bottom: 20px; }
.price-card--feature .price-card__desc { color: rgba(239, 234, 227, 0.6); }
.price { display: flex; align-items: baseline; gap: 6px; margin-bottom: 22px; padding-bottom: 22px; border-bottom: 1px solid var(--line); }
.price-card--feature .price { border-bottom-color: rgba(239, 234, 227, 0.16); }
.price__num { font-size: 2.5rem; font-weight: 800; letter-spacing: -0.02em; color: var(--char); }
.price-card--feature .price__num { color: var(--amber); }
.price__num::before { content: "$"; font-size: 1.3rem; font-weight: 600; vertical-align: super; margin-right: 1px; }
.price__per { font-size: 0.82rem; color: var(--muted); }
.price-card--feature .price__per { color: rgba(239, 234, 227, 0.6); }
.price-feats { list-style: none; display: grid; gap: 11px; margin-bottom: 26px; }
.price-feats li { font-size: 0.9rem; color: var(--ink-2); padding-left: 24px; position: relative; }
.price-card--feature .price-feats li { color: rgba(239, 234, 227, 0.82); }
.price-feats li::before { content: "✓"; position: absolute; left: 0; color: var(--plant); font-weight: 800; }
.price-card--feature .price-feats li::before { color: var(--amber); }
/* ---------- tour ---------- */
.tour { background: var(--amber-50); }
.tour__grid { display: grid; grid-template-columns: 1fr 1fr; gap: 54px; align-items: center; }
.tour__copy h2 { font-size: clamp(1.7rem, 3.6vw, 2.3rem); font-weight: 800; letter-spacing: -0.02em; color: var(--char); margin-bottom: 14px; line-height: 1.14; }
.tour__copy > p { color: var(--ink-2); font-size: 1.04rem; margin-bottom: 22px; }
.tour__points { list-style: none; display: grid; gap: 12px; }
.tour__points li { font-size: 0.96rem; color: var(--ink); font-weight: 500; }
.tour__points li::first-letter { color: var(--amber-d); }
.tour__form { background: var(--surface); border: 1px solid var(--line); border-radius: var(--r-lg); padding: 30px; box-shadow: var(--shadow-lg); display: grid; gap: 16px; }
.field { display: grid; gap: 7px; }
.field label { font-size: 0.84rem; font-weight: 600; color: var(--ink); }
.field input, .field select {
font: inherit; font-size: 0.95rem; padding: 12px 14px; border: 1.5px solid var(--line-2);
border-radius: var(--r-sm); background: var(--bg); color: var(--ink); width: 100%;
transition: border-color 0.18s ease, box-shadow 0.18s ease;
}
.field input:focus, .field select:focus { outline: none; border-color: var(--amber); box-shadow: 0 0 0 3px rgba(232, 144, 43, 0.18); background: var(--surface); }
.field input.is-error, .field select.is-error { border-color: var(--danger); box-shadow: 0 0 0 3px rgba(212, 80, 62, 0.15); }
.day-pick { display: grid; grid-template-columns: repeat(4, 1fr); gap: 8px; }
.day-pick button { font: inherit; font-weight: 600; font-size: 0.88rem; padding: 10px; border: 1.5px solid var(--line-2); border-radius: var(--r-sm); background: var(--bg); color: var(--ink-2); cursor: pointer; transition: all 0.16s ease; }
.day-pick button:hover { border-color: var(--char); color: var(--char); }
.day-pick button.is-active { background: var(--char); color: #fff; border-color: var(--char); }
.day-pick button:focus-visible { outline: 3px solid var(--amber); outline-offset: 2px; }
.tour__fineprint { font-size: 0.78rem; color: var(--muted); text-align: center; }
/* ---------- footer ---------- */
.footer { background: var(--char); color: var(--concrete); padding: 56px 0 28px; }
.footer__grid { display: grid; grid-template-columns: 1.8fr 1fr 1fr 1.1fr; gap: 36px; padding-bottom: 38px; border-bottom: 1px solid rgba(239, 234, 227, 0.12); }
.footer__brand .brand { margin-bottom: 16px; }
.footer__brand .brand__name { color: var(--concrete); }
.footer__brand p { font-size: 0.9rem; color: rgba(239, 234, 227, 0.6); max-width: 38ch; margin-bottom: 16px; }
.footer__plant { font-size: 0.84rem; color: var(--plant); font-weight: 600; }
.footer__col h4 { font-size: 0.8rem; letter-spacing: 0.1em; text-transform: uppercase; color: rgba(239, 234, 227, 0.5); margin-bottom: 16px; }
.footer__col a { display: block; font-size: 0.92rem; color: rgba(239, 234, 227, 0.78); padding: 6px 0; transition: color 0.16s ease, padding-left 0.16s ease; }
.footer__col a:hover { color: var(--amber); padding-left: 5px; }
.footer__hours p { font-size: 0.88rem; color: rgba(239, 234, 227, 0.7); padding: 4px 0; }
.footer__base { display: flex; justify-content: space-between; gap: 16px; padding-top: 22px; flex-wrap: wrap; }
.footer__base span { font-size: 0.82rem; color: rgba(239, 234, 227, 0.5); }
.footer__made { color: rgba(232, 144, 43, 0.7) !important; }
/* ---------- toast ---------- */
.toast {
position: fixed; left: 50%; bottom: 28px; transform: translate(-50%, 120%);
background: var(--char); color: var(--concrete); padding: 13px 22px; border-radius: 100px;
font-size: 0.9rem; font-weight: 600; box-shadow: var(--shadow-lg); z-index: 100;
opacity: 0; transition: transform 0.36s cubic-bezier(0.2, 0.9, 0.3, 1.2), opacity 0.3s ease;
max-width: calc(100vw - 40px); text-align: center;
}
.toast.is-show { transform: translate(-50%, 0); opacity: 1; }
.toast::before { content: "◆ "; color: var(--amber); }
/* ---------- reveal ---------- */
.reveal { opacity: 0; transform: translateY(26px); transition: opacity 0.6s ease, transform 0.6s cubic-bezier(0.2, 0.7, 0.3, 1); }
.reveal.is-in { opacity: 1; transform: none; }
@media (prefers-reduced-motion: reduce) {
.reveal { opacity: 1; transform: none; transition: none; }
.marquee__track { animation: none; }
html { scroll-behavior: auto; }
}
/* ---------- responsive ---------- */
@media (max-width: 940px) {
.hero__grid { grid-template-columns: 1fr; gap: 40px; }
.hero__lede { max-width: none; }
.who-grid { grid-template-columns: repeat(2, 1fr); }
.amen-grid { grid-template-columns: repeat(2, 1fr); }
.makers__grid, .tour__grid { grid-template-columns: 1fr; gap: 36px; }
.price-grid { grid-template-columns: 1fr; max-width: 440px; margin: 0 auto; }
.price-card--feature { transform: none; }
.price-card--feature:hover { transform: translateY(-5px); }
.footer__grid { grid-template-columns: 1fr 1fr; gap: 30px; }
.nav__links, .nav__cta { display: none; }
.nav__burger { display: flex; }
}
@media (max-width: 520px) {
.wrap { padding: 0 18px; }
.section { padding: 60px 0; }
.hero { padding: 44px 0 40px; }
.who-grid, .amen-grid { grid-template-columns: 1fr; }
.hero__stats { gap: 22px; }
.hero__stats dt { font-size: 1.6rem; }
.footer__grid { grid-template-columns: 1fr; }
.hero__art { grid-template-columns: 1fr 1fr; }
.photo--tall { min-height: 240px; }
.btn--lg { width: 100%; }
.hero__actions { flex-direction: column; }
.day-pick { grid-template-columns: repeat(2, 1fr); }
.footer__base { flex-direction: column; gap: 6px; }
}(function () {
"use strict";
/* ---------- toast helper ---------- */
var toastEl = document.getElementById("toast");
var toastTimer;
function toast(msg) {
if (!toastEl) return;
toastEl.textContent = msg;
toastEl.classList.add("is-show");
clearTimeout(toastTimer);
toastTimer = setTimeout(function () {
toastEl.classList.remove("is-show");
}, 2800);
}
/* ---------- sticky nav shadow ---------- */
var nav = document.getElementById("nav");
function onScroll() {
if (window.scrollY > 8) nav.classList.add("is-scrolled");
else nav.classList.remove("is-scrolled");
}
window.addEventListener("scroll", onScroll, { passive: true });
onScroll();
/* ---------- mobile nav ---------- */
var burger = document.getElementById("burger");
var mobileNav = document.getElementById("mobileNav");
function closeMenu() {
burger.classList.remove("is-open");
mobileNav.classList.remove("is-open");
burger.setAttribute("aria-expanded", "false");
}
burger.addEventListener("click", function () {
var open = mobileNav.classList.toggle("is-open");
burger.classList.toggle("is-open", open);
burger.setAttribute("aria-expanded", String(open));
});
mobileNav.querySelectorAll("a").forEach(function (a) {
a.addEventListener("click", closeMenu);
});
/* ---------- scroll reveal ---------- */
var reveals = document.querySelectorAll(".reveal");
if ("IntersectionObserver" in window) {
var io = new IntersectionObserver(
function (entries) {
entries.forEach(function (e) {
if (e.isIntersecting) {
e.target.classList.add("is-in");
io.unobserve(e.target);
}
});
},
{ threshold: 0.12, rootMargin: "0px 0px -8% 0px" }
);
reveals.forEach(function (el) {
io.observe(el);
});
} else {
reveals.forEach(function (el) {
el.classList.add("is-in");
});
}
/* ---------- hero stat count-up ---------- */
var stats = [
{ id: "statMembers", to: 96, suffix: "" },
{ id: "statSqft", to: 12000, suffix: "" },
{ id: "statYears", to: 8, suffix: "" }
];
function fmt(n) {
return n.toLocaleString("en-US");
}
function countUp(el, to) {
var start = performance.now();
var dur = 1400;
function step(now) {
var p = Math.min((now - start) / dur, 1);
var eased = 1 - Math.pow(1 - p, 3);
el.textContent = fmt(Math.round(to * eased));
if (p < 1) requestAnimationFrame(step);
}
requestAnimationFrame(step);
}
var heroSeen = false;
function maybeCount() {
if (heroSeen) return;
var hero = document.getElementById("hero");
if (hero.getBoundingClientRect().top < window.innerHeight) {
heroSeen = true;
stats.forEach(function (s) {
var el = document.getElementById(s.id);
if (el) countUp(el, s.to);
});
}
}
maybeCount();
window.addEventListener("scroll", maybeCount, { passive: true });
/* ---------- amenities ---------- */
var amenities = [
{ icon: "🖨", name: "Riso print studio", desc: "2-color GR3770 + drying rack", status: "free", label: "Open now" },
{ icon: "📷", name: "Photo bay & cyc wall", desc: "Blackout, 3 strobes, tethered", status: "busy", label: "In use · til 4pm" },
{ icon: "🪵", name: "Dust-walled woodshop", desc: "Table saw, sander, 12 clamps", status: "free", label: "Open now" },
{ icon: "🔥", name: "Twin ceramic kilns", desc: "Bisque + glaze, 1280°C", status: "soon", label: "Firing 9am" },
{ icon: "🎞", name: "Wet darkroom", desc: "B&W, 2 enlargers, chem stocked", status: "free", label: "Open now" },
{ icon: "☕", name: "Loft café & cold brew", desc: "Bottomless · roasted in-house", status: "free", label: "Always on" }
];
var statusMap = {
free: { cls: "is-free", booked: "Booked you a slot — fob will glow amber when it's yours." },
busy: { cls: "is-busy", booked: "Added you to the waitlist — we'll ping you when it frees up." },
soon: { cls: "is-soon", booked: "Reserved your turn after the current firing finishes." }
};
var amenGrid = document.getElementById("amenGrid");
if (amenGrid) {
amenities.forEach(function (a) {
var s = statusMap[a.status];
var card = document.createElement("button");
card.type = "button";
card.className = "amen-card";
card.innerHTML =
'<div class="amen-card__head">' +
'<span class="amen-card__icon" aria-hidden="true">' + a.icon + "</span>" +
'<span class="amen-status ' + s.cls + '">' + a.label + "</span>" +
"</div>" +
"<h3>" + a.name + "</h3>" +
"<p>" + a.desc + "</p>";
card.setAttribute("aria-label", a.name + " — " + a.label + ". Tap to book.");
card.addEventListener("click", function () {
toast(a.name + " · " + s.booked);
});
amenGrid.appendChild(card);
});
}
/* ---------- pricing toggle ---------- */
var billBtns = document.querySelectorAll(".bill-toggle__btn");
var priceNums = document.querySelectorAll(".price__num");
billBtns.forEach(function (btn) {
btn.addEventListener("click", function () {
var mode = btn.getAttribute("data-bill");
billBtns.forEach(function (b) {
var on = b === btn;
b.classList.toggle("is-active", on);
b.setAttribute("aria-pressed", String(on));
});
priceNums.forEach(function (n) {
var val = n.getAttribute(mode === "annual" ? "data-annual" : "data-monthly");
if (val) n.textContent = val;
});
toast(mode === "annual" ? "Annual pricing — 15% off, billed yearly." : "Switched to monthly pricing.");
});
});
/* ---------- plan buttons ---------- */
document.querySelectorAll("[data-plan]").forEach(function (b) {
b.addEventListener("click", function () {
var plan = b.getAttribute("data-plan");
toast(plan + " selected — let's book your tour to lock it in.");
var tour = document.getElementById("tour");
if (tour) tour.scrollIntoView({ behavior: "smooth", block: "start" });
});
});
/* ---------- day picker ---------- */
var dayPick = document.getElementById("dayPick");
var dayInput = document.getElementById("tDay");
if (dayPick) {
dayPick.querySelectorAll("button").forEach(function (b) {
b.addEventListener("click", function () {
dayPick.querySelectorAll("button").forEach(function (x) {
x.classList.remove("is-active");
});
b.classList.add("is-active");
dayInput.value = b.getAttribute("data-day");
});
});
}
/* ---------- tour form ---------- */
var form = document.getElementById("tourForm");
if (form) {
var emailRe = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
form.addEventListener("submit", function (e) {
e.preventDefault();
var name = form.querySelector("#tName");
var email = form.querySelector("#tEmail");
var craft = form.querySelector("#tCraft");
[name, email, craft].forEach(function (f) {
f.classList.remove("is-error");
});
var bad = null;
if (!name.value.trim()) bad = name;
else if (!emailRe.test(email.value.trim())) bad = email;
else if (!craft.value) bad = craft;
if (bad) {
bad.classList.add("is-error");
bad.focus();
toast("Just need a name, a valid email and your craft.");
return;
}
if (!dayInput.value) {
toast("Pick a preferred day for your tour.");
dayPick.querySelector("button").focus();
return;
}
toast("Tour requested for " + dayInput.value + " — see you at Warehouse 14, " + name.value.trim().split(" ")[0] + "!");
form.reset();
dayPick.querySelectorAll("button").forEach(function (x) {
x.classList.remove("is-active");
});
dayInput.value = "";
});
}
/* ---------- live amenity drift (subtle) ---------- */
setInterval(function () {
var cards = document.querySelectorAll(".amen-status.is-busy");
if (!cards.length) return;
// gentle pulse on a busy amenity to feel alive
var pick = cards[Math.floor(Math.random() * cards.length)];
pick.animate(
[{ opacity: 1 }, { opacity: 0.55 }, { opacity: 1 }],
{ duration: 900, easing: "ease-in-out" }
);
}, 5000);
})();<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>Foundry Loft — Creative Coworking Studio</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>
<!-- ============ NAV ============ -->
<header class="nav" id="nav">
<div class="wrap nav__inner">
<a class="brand" href="#top" aria-label="Foundry Loft home">
<span class="brand__mark" aria-hidden="true">◢◣</span>
<span class="brand__name">FOUNDRY<span class="brand__sub">LOFT</span></span>
</a>
<nav class="nav__links" aria-label="Primary">
<a href="#who">Who it's for</a>
<a href="#studio">The studio</a>
<a href="#makers">Makers</a>
<a href="#pricing">Pricing</a>
</nav>
<div class="nav__cta">
<a href="#tour" class="btn btn--ghost">Book a tour</a>
</div>
<button class="nav__burger" id="burger" aria-label="Open menu" aria-expanded="false" aria-controls="mobileNav">
<span></span><span></span><span></span>
</button>
</div>
<nav class="mobile-nav" id="mobileNav" aria-label="Mobile">
<a href="#who">Who it's for</a>
<a href="#studio">The studio</a>
<a href="#makers">Makers</a>
<a href="#pricing">Pricing</a>
<a href="#tour" class="btn btn--amber">Book a tour</a>
</nav>
</header>
<main id="top">
<!-- ============ HERO ============ -->
<section class="hero" id="hero">
<div class="wrap hero__grid">
<div class="hero__copy reveal">
<span class="eyebrow">Warehouse no. 14 · East Foundry District</span>
<h1>A raw loft built for people who <em>make things.</em></h1>
<p class="hero__lede">
12,000 sq ft of concrete, north light and amber afternoons — shared by
designers, illustrators, photographers and the stubbornly self-employed.
No cubicles. No fluorescent ceiling tiles. Just good light and good company.
</p>
<div class="hero__actions">
<a href="#tour" class="btn btn--amber btn--lg">Book a studio tour</a>
<a href="#studio" class="btn btn--line btn--lg">See the space</a>
</div>
<dl class="hero__stats">
<div><dt id="statMembers">0</dt><dd>resident makers</dd></div>
<div><dt id="statSqft">0</dt><dd>sq ft of loft</dd></div>
<div><dt id="statYears">0</dt><dd>years in the district</dd></div>
</dl>
</div>
<div class="hero__art reveal" aria-hidden="true">
<div class="photo photo--tall" data-label="Main floor · north windows"></div>
<div class="photo photo--amber" data-label="Print studio"></div>
<div class="photo photo--char" data-label="The mezzanine"></div>
<span class="art__tag">est. 2018</span>
</div>
</div>
</section>
<!-- ============ MARQUEE ============ -->
<div class="marquee" aria-hidden="true">
<div class="marquee__track">
<span>RISO PRINTING</span><span>◆</span><span>PHOTO BAY</span><span>◆</span>
<span>WOODSHOP</span><span>◆</span><span>NORTH LIGHT</span><span>◆</span>
<span>OPEN STUDIO</span><span>◆</span><span>DARKROOM</span><span>◆</span>
<span>RISO PRINTING</span><span>◆</span><span>PHOTO BAY</span><span>◆</span>
<span>WOODSHOP</span><span>◆</span><span>NORTH LIGHT</span><span>◆</span>
<span>OPEN STUDIO</span><span>◆</span><span>DARKROOM</span><span>◆</span>
</div>
</div>
<!-- ============ WHO IT'S FOR ============ -->
<section class="section" id="who">
<div class="wrap">
<header class="sec-head reveal">
<span class="kicker">Who it's for</span>
<h2>Built around the way creative people actually work.</h2>
<p>Heads-down for six hours, then loud and collaborative for one. Messy
desks allowed. Big prints encouraged.</p>
</header>
<div class="who-grid">
<article class="who-card reveal" tabindex="0">
<span class="who-card__icon" aria-hidden="true">✎</span>
<h3>Designers & illustrators</h3>
<p>Dual-monitor desks, color-true north light, and a riso for proofs that
never quite look right on screen.</p>
<span class="who-card__tag">42 residents</span>
</article>
<article class="who-card reveal" tabindex="0">
<span class="who-card__icon" aria-hidden="true">◎</span>
<h3>Photographers</h3>
<p>A bookable cyc wall, blackout photo bay and a wet darkroom for the
film holdouts among us.</p>
<span class="who-card__tag">18 residents</span>
</article>
<article class="who-card reveal" tabindex="0">
<span class="who-card__icon" aria-hidden="true">⚒</span>
<h3>Makers & ceramicists</h3>
<p>A dust-walled woodshop, two kilns and bench space rated for the kind of
mess landlords hate.</p>
<span class="who-card__tag">11 residents</span>
</article>
<article class="who-card reveal" tabindex="0">
<span class="who-card__icon" aria-hidden="true">⌥</span>
<h3>Freelancers & studios</h3>
<p>Private 2–6 person studios with roller doors, plus drop-in credits for
the days you just need to get out of the house.</p>
<span class="who-card__tag">14 studios</span>
</article>
</div>
</div>
</section>
<!-- ============ STUDIO AMENITIES ============ -->
<section class="section section--char" id="studio">
<div class="wrap">
<header class="sec-head sec-head--light reveal">
<span class="kicker kicker--amber">The studio</span>
<h2>Everything a home office can't give you.</h2>
<p>Shared gear that's too big, too loud or too expensive to keep at home —
maintained, stocked, and ready when you are.</p>
</header>
<div class="amen-grid" id="amenGrid">
<!-- injected -->
</div>
<p class="amen-note reveal">Tap an amenity to check today's availability.</p>
</div>
</section>
<!-- ============ COMMUNITY OF MAKERS ============ -->
<section class="section" id="makers">
<div class="wrap makers__grid">
<div class="makers__copy reveal">
<span class="kicker">Community of makers</span>
<h2>You'll borrow a clamp and leave with a collaborator.</h2>
<p>Foundry Loft runs on the small generosities of people who share walls:
a borrowed lens, a second opinion on a logo, a critique over cold brew.
Wednesday crits, Friday show-and-tell, and a members' wall that's never empty.</p>
<ul class="makers__list">
<li><strong>Weekly crits</strong> — open desk reviews, no ego required</li>
<li><strong>Skill swaps</strong> — trade a retouch for a typeface</li>
<li><strong>Open studios</strong> — quarterly public nights with sales</li>
<li><strong>Members' board</strong> — gigs, gear and gallery calls</li>
</ul>
<div class="makers__avatars" aria-label="Some of our resident makers">
<span class="avi avi--a" title="Mara Okafor, illustrator">MO</span>
<span class="avi avi--b" title="Dev Sandoval, photographer">DS</span>
<span class="avi avi--c" title="Lena Brandt, ceramicist">LB</span>
<span class="avi avi--d" title="Theo Park, brand designer">TP</span>
<span class="avi avi--e" title="Ines Vidal, animator">IV</span>
<span class="avi avi--more">+78</span>
</div>
</div>
<div class="makers__feed reveal" aria-label="Recent community posts">
<div class="feed-card">
<div class="feed-card__top">
<span class="avi avi--a">MO</span>
<div><strong>Mara Okafor</strong><span>Illustrator · 2h</span></div>
</div>
<p>Riso's loaded with fluorescent pink + black until Friday — come grab proofs.</p>
<div class="feed-card__meta">🟢 Print studio · 14 ♥</div>
</div>
<div class="feed-card">
<div class="feed-card__top">
<span class="avi avi--b">DS</span>
<div><strong>Dev Sandoval</strong><span>Photographer · 5h</span></div>
</div>
<p>Cyc wall free this aft if anyone needs product shots. Bringing the strobes.</p>
<div class="feed-card__meta">🟢 Photo bay · 9 ♥</div>
</div>
<div class="feed-card">
<div class="feed-card__top">
<span class="avi avi--c">LB</span>
<div><strong>Lena Brandt</strong><span>Ceramicist · 1d</span></div>
</div>
<p>Bisque firing tomorrow 9am — pop your greenware on the kiln shelf tonight.</p>
<div class="feed-card__meta">🔥 Woodshop · 21 ♥</div>
</div>
</div>
</div>
</section>
<!-- ============ PRICING ============ -->
<section class="section section--concrete" id="pricing">
<div class="wrap">
<header class="sec-head reveal">
<span class="kicker">Membership</span>
<h2>Pick the amount of loft you need.</h2>
<p>Monthly, no contracts. Switch tiers or pause any time — we get that
freelance income comes in waves.</p>
</header>
<div class="bill-toggle reveal" role="group" aria-label="Billing period">
<button class="bill-toggle__btn is-active" data-bill="monthly" aria-pressed="true">Monthly</button>
<button class="bill-toggle__btn" data-bill="annual" aria-pressed="false">Annual <span>−15%</span></button>
</div>
<div class="price-grid">
<article class="price-card reveal">
<h3>Drop-in</h3>
<p class="price-card__desc">For the out-of-the-house days.</p>
<div class="price"><span class="price__num" data-monthly="49" data-annual="42">49</span><span class="price__per">/mo · 8 day credits</span></div>
<ul class="price-feats">
<li>8 day passes a month</li>
<li>Any open hot desk</li>
<li>Coffee, wifi, north light</li>
<li>Community board access</li>
</ul>
<button class="btn btn--line btn--block" data-plan="Drop-in">Start drop-in</button>
</article>
<article class="price-card price-card--feature reveal">
<span class="price-card__flag">Most makers pick this</span>
<h3>Resident desk</h3>
<p class="price-card__desc">Your own desk, your own mess.</p>
<div class="price"><span class="price__num" data-monthly="285" data-annual="242">285</span><span class="price__per">/mo · dedicated</span></div>
<ul class="price-feats">
<li>A desk that stays yours</li>
<li>24/7 amber-fob access</li>
<li>6 hrs shared gear / week</li>
<li>Storage locker + crit nights</li>
<li>2 meeting-room hrs / week</li>
</ul>
<button class="btn btn--amber btn--block" data-plan="Resident desk">Claim a desk</button>
</article>
<article class="price-card reveal">
<h3>Private studio</h3>
<p class="price-card__desc">A room with a roller door.</p>
<div class="price"><span class="price__num" data-monthly="690" data-annual="586">690</span><span class="price__per">/mo · 2–6 people</span></div>
<ul class="price-feats">
<li>Lockable private studio</li>
<li>Up to 6 member fobs</li>
<li>Unlimited shared gear</li>
<li>Name on the studio door</li>
<li>Priority open-studio wall</li>
</ul>
<button class="btn btn--line btn--block" data-plan="Private studio">Enquire on studios</button>
</article>
</div>
</div>
</section>
<!-- ============ BOOK A TOUR ============ -->
<section class="section tour" id="tour">
<div class="wrap tour__grid">
<div class="tour__copy reveal">
<span class="kicker kicker--amber">Come see it</span>
<h2>Book a tour. Bring your messiest project.</h2>
<p>We'll walk you through the floor, fire up the riso, and let you sit at a
desk for an hour to feel the light. Tours run Tue–Fri.</p>
<ul class="tour__points">
<li>✦ 30-minute walk-through with a resident</li>
<li>✦ A free day pass to actually try it</li>
<li>✦ No pressure, no sales script</li>
</ul>
</div>
<form class="tour__form reveal" id="tourForm" novalidate>
<div class="field">
<label for="tName">Name</label>
<input id="tName" name="name" type="text" autocomplete="name" placeholder="Robin Vega" required />
</div>
<div class="field">
<label for="tEmail">Email</label>
<input id="tEmail" name="email" type="email" autocomplete="email" placeholder="robin@studio.com" required />
</div>
<div class="field">
<label for="tCraft">What do you make?</label>
<select id="tCraft" name="craft" required>
<option value="">Pick one…</option>
<option>Design / illustration</option>
<option>Photography</option>
<option>Ceramics / woodwork</option>
<option>Film / animation</option>
<option>Something else</option>
</select>
</div>
<div class="field">
<label for="tDay">Preferred day</label>
<div class="day-pick" id="dayPick" role="group" aria-label="Preferred tour day">
<button type="button" data-day="Tue">Tue</button>
<button type="button" data-day="Wed">Wed</button>
<button type="button" data-day="Thu">Thu</button>
<button type="button" data-day="Fri">Fri</button>
</div>
<input type="hidden" id="tDay" name="day" />
</div>
<button type="submit" class="btn btn--amber btn--block btn--lg">Request my tour</button>
<p class="tour__fineprint">We'll reply within a working day. No spam, ever.</p>
</form>
</div>
</section>
</main>
<!-- ============ FOOTER ============ -->
<footer class="footer">
<div class="wrap footer__grid">
<div class="footer__brand">
<a class="brand" href="#top">
<span class="brand__mark" aria-hidden="true">◢◣</span>
<span class="brand__name">FOUNDRY<span class="brand__sub">LOFT</span></span>
</a>
<p>Warehouse no. 14, East Foundry District. A creative loft for people who
make things with their hands and their software.</p>
<div class="footer__plant" aria-hidden="true">🪴 — yes, the plants are real.</div>
</div>
<div class="footer__col">
<h4>Studio</h4>
<a href="#who">Who it's for</a>
<a href="#studio">Amenities</a>
<a href="#makers">Community</a>
<a href="#pricing">Pricing</a>
</div>
<div class="footer__col">
<h4>Visit</h4>
<a href="#tour">Book a tour</a>
<a href="#top">Open studios</a>
<a href="#top">Members' board</a>
<a href="#top">Contact</a>
</div>
<div class="footer__col footer__hours">
<h4>Hours</h4>
<p>Members · 24/7 fob access</p>
<p>Reception · Mon–Fri, 9–6</p>
<p>Tours · Tue–Fri, by appt.</p>
</div>
</div>
<div class="wrap footer__base">
<span>© 2026 Foundry Loft — a fictional space.</span>
<span class="footer__made">Made with concrete, amber & north light.</span>
</div>
</footer>
<div class="toast" id="toast" role="status" aria-live="polite"></div>
<script src="script.js"></script>
</body>
</html>Creative Loft Landing
A full marketing landing page for Foundry Loft, a fictional warehouse coworking studio built for designers, illustrators, photographers and makers. The raw industrial hero sits over a faded concrete grid with amber-underlined headline accents, warm gradient photo blocks standing in for loft photography, and three hero stats that count up as the page loads. A sticky nav fades in a shadow on scroll and collapses into an animated burger menu on small screens.
Below the hero, an infinite amenities marquee leads into a “who it’s for” card grid, a dark studio-amenities section where each card carries a free / in-use / firing-soon status chip and books a slot with toast feedback, and a community section that mixes overlapping member avatars with a live-feeling post feed. The pricing block flips between monthly and annual rates from a single toggle, and the book-a-tour form validates name, email, craft and a day picker before confirming with a personalised toast.
Everything is vanilla HTML, CSS and JavaScript — no frameworks or build step. Sections reveal on scroll via IntersectionObserver, the layout reflows cleanly down to 360px, and all controls are keyboard-usable with visible focus rings and AA-contrast text.
Illustrative UI only — fictional coworking space, not a real booking system.