Web3 — NFT Project / Collection Landing
A hype, glow-heavy landing page for a fictional generative PFP drop called VOIDLINGS. Features a glitching hero, an animated marquee of CSS-drawn generative avatars, and a full mint panel with a live countdown, minted-versus-total progress bar, a quantity stepper, and a simulated mint flow with confirm and success states. Rounds out with lore, a hover-to-reveal rarity and traits grid, a roadmap timeline, a team grid, an accordion FAQ, and a community CTA. Vanilla JS, no libraries.
MCP
Code
:root {
/* NFT palette override — hype, glow */
--bg: #0d0618;
--surface: #160a26;
--surface-2: #1d1033;
--elevated: #251642;
--text: #f3ecff;
--muted: #9b8ec2;
--line: rgba(255, 255, 255, 0.08);
--line-2: rgba(255, 255, 255, 0.16);
--accent: #ff4d8d; /* hot pink */
--accent-2: #3fd0ff; /* cyan */
--accent-3: #b86bff; /* violet bridge */
--accent-glow: rgba(255, 77, 141, 0.5);
--cyan-glow: rgba(63, 208, 255, 0.45);
--pos: #26d07c;
--neg: #ff4d6d;
--warn: #ffb347;
--r-sm: 8px;
--r-md: 14px;
--r-lg: 20px;
--r-pill: 999px;
--grad: linear-gradient(120deg, var(--accent), var(--accent-3) 55%, var(--accent-2));
--font: "Space Grotesk", system-ui, sans-serif;
--mono: "JetBrains Mono", ui-monospace, monospace;
}
* { box-sizing: border-box; }
html { scroll-behavior: smooth; }
body {
margin: 0;
font-family: var(--font);
background: var(--bg);
color: var(--text);
line-height: 1.5;
-webkit-font-smoothing: antialiased;
text-rendering: optimizeLegibility;
overflow-x: hidden;
}
.mono { font-family: var(--mono); font-variant-numeric: tabular-nums; }
.muted { color: var(--muted); }
a { color: inherit; text-decoration: none; }
h1, h2, h3 { margin: 0; line-height: 1.1; letter-spacing: -0.02em; }
:focus-visible {
outline: 2px solid var(--accent-2);
outline-offset: 3px;
border-radius: 4px;
}
/* ===== ambient aurora ===== */
.aurora {
position: fixed;
inset: -20% -20% auto -20%;
height: 70vh;
z-index: 0;
pointer-events: none;
background:
radial-gradient(50% 60% at 20% 20%, var(--accent-glow), transparent 60%),
radial-gradient(45% 55% at 85% 10%, var(--cyan-glow), transparent 60%),
radial-gradient(40% 50% at 60% 80%, rgba(184, 107, 255, 0.35), transparent 60%);
filter: blur(20px);
opacity: 0.85;
animation: drift 16s ease-in-out infinite alternate;
}
@keyframes drift {
from { transform: translate3d(-3%, -2%, 0) scale(1); }
to { transform: translate3d(4%, 3%, 0) scale(1.12); }
}
/* ===== layout ===== */
.nav, .hero, .section, .community-inner, .footer {
width: min(1120px, 92vw);
margin-inline: auto;
position: relative;
z-index: 1;
}
/* ===== nav ===== */
.nav {
display: flex;
align-items: center;
gap: 1.5rem;
padding: 1.1rem 0;
position: sticky;
top: 0;
z-index: 50;
}
.nav::before {
content: "";
position: absolute;
inset: 0 -50vw;
background: rgba(13, 6, 24, 0.55);
backdrop-filter: blur(14px);
border-bottom: 1px solid var(--line);
z-index: -1;
}
.brand { display: flex; align-items: center; gap: 0.6rem; font-weight: 700; }
.brand-mark {
width: 26px; height: 26px; border-radius: 8px;
background: var(--grad);
box-shadow: 0 0 18px var(--accent-glow);
animation: spin 9s linear infinite;
}
@keyframes spin { to { transform: rotate(360deg); } }
.brand-name { letter-spacing: 0.18em; font-size: 1.05rem; }
.nav-links { display: flex; gap: 1.3rem; margin-left: auto; font-size: 0.92rem; }
.nav-links a { color: var(--muted); transition: color 0.18s; }
.nav-links a:hover { color: var(--text); }
/* ===== buttons ===== */
.btn {
font-family: var(--font);
font-weight: 600;
font-size: 0.95rem;
border: 0;
border-radius: var(--r-pill);
padding: 0.7rem 1.3rem;
cursor: pointer;
display: inline-flex;
align-items: center;
justify-content: center;
gap: 0.45rem;
transition: transform 0.15s, box-shadow 0.2s, background 0.2s, border-color 0.2s;
color: var(--text);
}
.btn:active { transform: translateY(1px) scale(0.99); }
.btn-ghost {
background: rgba(255, 255, 255, 0.04);
border: 1px solid var(--line-2);
}
.btn-ghost:hover { background: rgba(255, 255, 255, 0.09); border-color: var(--accent-2); }
.btn-primary {
background: var(--grad);
color: #fff;
box-shadow: 0 8px 28px var(--accent-glow);
}
.btn-primary:hover { box-shadow: 0 10px 38px var(--accent-glow); transform: translateY(-2px); }
.connect { margin-left: 0; }
/* ===== hero ===== */
.hero {
display: grid;
grid-template-columns: 1.15fr 0.85fr;
gap: 2.5rem;
align-items: center;
padding: clamp(2.5rem, 7vw, 5.5rem) 0 3rem;
}
.pill {
display: inline-flex;
align-items: center;
gap: 0.5rem;
font-size: 0.8rem;
font-weight: 500;
color: var(--accent-2);
background: rgba(63, 208, 255, 0.1);
border: 1px solid rgba(63, 208, 255, 0.3);
padding: 0.35rem 0.85rem;
border-radius: var(--r-pill);
}
.dot {
width: 7px; height: 7px; border-radius: 50%;
background: var(--accent-2);
box-shadow: 0 0 10px var(--accent-2);
animation: pulse 1.4s ease-in-out infinite;
}
@keyframes pulse { 0%, 100% { opacity: 1; } 50% { opacity: 0.35; } }
.hero-title {
font-size: clamp(3rem, 11vw, 6.5rem);
font-weight: 700;
margin: 1.1rem 0 0.9rem;
letter-spacing: -0.03em;
}
.glitch {
position: relative;
display: inline-block;
background: var(--grad);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
filter: drop-shadow(0 0 24px var(--accent-glow));
}
.glitch::before, .glitch::after {
content: attr(data-text);
position: absolute;
inset: 0;
background: var(--bg);
-webkit-background-clip: initial;
background-clip: initial;
clip-path: inset(0 0 0 0);
}
.glitch::before {
color: var(--accent-2);
animation: glitch1 3.4s infinite steps(2, end);
mix-blend-mode: screen;
background: transparent;
}
.glitch::after {
color: var(--accent);
animation: glitch2 2.7s infinite steps(2, end);
mix-blend-mode: screen;
background: transparent;
}
@keyframes glitch1 {
0%, 92%, 100% { transform: none; clip-path: inset(0 0 100% 0); opacity: 0; }
93% { transform: translate(-3px, 1px); clip-path: inset(10% 0 60% 0); opacity: 0.8; }
96% { transform: translate(2px, -2px); clip-path: inset(40% 0 30% 0); opacity: 0.8; }
}
@keyframes glitch2 {
0%, 90%, 100% { transform: none; clip-path: inset(0 0 100% 0); opacity: 0; }
91% { transform: translate(3px, -1px); clip-path: inset(60% 0 10% 0); opacity: 0.7; }
94% { transform: translate(-2px, 2px); clip-path: inset(20% 0 55% 0); opacity: 0.7; }
}
.hero-sub { color: var(--muted); font-size: 1.1rem; max-width: 44ch; margin: 0 0 1.8rem; }
.hero-stats { display: flex; gap: 2rem; margin-bottom: 1.8rem; flex-wrap: wrap; }
.stat { display: flex; flex-direction: column; }
.stat-num { font-family: var(--mono); font-size: 1.7rem; font-weight: 700; }
.stat-label { font-size: 0.78rem; color: var(--muted); text-transform: uppercase; letter-spacing: 0.08em; }
.hero-cta { display: flex; gap: 0.8rem; flex-wrap: wrap; }
/* ===== mint card ===== */
.mint-card {
position: relative;
border-radius: calc(var(--r-lg) + 2px);
padding: 2px;
background: var(--grad);
box-shadow: 0 24px 60px rgba(0, 0, 0, 0.5), 0 0 50px rgba(255, 77, 141, 0.18);
}
.mint-card-inner {
background: linear-gradient(180deg, rgba(37, 22, 66, 0.92), rgba(22, 10, 38, 0.95));
backdrop-filter: blur(16px);
border-radius: var(--r-lg);
padding: 1.6rem;
display: flex;
flex-direction: column;
gap: 1.25rem;
}
.mint-head { display: flex; justify-content: space-between; align-items: flex-start; gap: 1rem; }
.mint-title { font-size: 1.4rem; font-weight: 700; }
.mint-sub { color: var(--muted); font-size: 0.85rem; margin: 0.2rem 0 0; }
.badge-live {
display: inline-flex; align-items: center; gap: 0.4rem;
font-size: 0.72rem; font-weight: 700; letter-spacing: 0.1em;
color: var(--accent-2);
border: 1px solid rgba(63, 208, 255, 0.35);
border-radius: var(--r-pill);
padding: 0.3rem 0.7rem;
white-space: nowrap;
}
/* countdown */
.cd-label { font-size: 0.78rem; color: var(--muted); text-transform: uppercase; letter-spacing: 0.08em; }
.cd-grid { display: grid; grid-template-columns: repeat(4, 1fr); gap: 0.5rem; margin-top: 0.5rem; }
.cd-cell {
background: rgba(255, 255, 255, 0.04);
border: 1px solid var(--line);
border-radius: var(--r-md);
padding: 0.6rem 0.2rem;
text-align: center;
}
.cd-cell b {
display: block;
font-family: var(--mono);
font-size: 1.5rem;
font-weight: 700;
color: var(--text);
}
.cd-cell span { font-size: 0.65rem; color: var(--muted); text-transform: uppercase; letter-spacing: 0.06em; }
/* progress */
.progress-meta { display: flex; justify-content: space-between; font-size: 0.82rem; margin-bottom: 0.45rem; }
.progress {
height: 12px;
border-radius: var(--r-pill);
background: rgba(255, 255, 255, 0.06);
overflow: hidden;
border: 1px solid var(--line);
}
.progress-fill {
height: 100%;
width: 0%;
border-radius: var(--r-pill);
background: var(--grad);
box-shadow: 0 0 16px var(--accent-glow);
transition: width 0.6s cubic-bezier(0.22, 1, 0.36, 1);
}
/* qty / cost */
.qty-row, .cost-row { display: flex; align-items: center; justify-content: space-between; }
.qty-label { font-size: 0.92rem; }
.stepper {
display: flex; align-items: center; gap: 0.2rem;
background: rgba(255, 255, 255, 0.04);
border: 1px solid var(--line-2);
border-radius: var(--r-pill);
padding: 0.25rem;
}
.step-btn {
width: 34px; height: 34px;
border: 0; border-radius: 50%;
background: transparent; color: var(--text);
font-size: 1.3rem; line-height: 1; cursor: pointer;
transition: background 0.15s, transform 0.1s;
}
.step-btn:hover:not(:disabled) { background: rgba(255, 255, 255, 0.1); }
.step-btn:active:not(:disabled) { transform: scale(0.9); }
.step-btn:disabled { opacity: 0.3; cursor: not-allowed; }
.qty-val { min-width: 2ch; text-align: center; font-size: 1.1rem; font-weight: 700; }
.cost { font-size: 1.25rem; font-weight: 700; color: var(--accent-2); }
.btn-mint {
width: 100%;
background: var(--grad);
color: #fff;
font-size: 1.05rem;
padding: 0.95rem;
box-shadow: 0 10px 32px var(--accent-glow);
position: relative;
overflow: hidden;
}
.btn-mint:hover:not(:disabled) { box-shadow: 0 14px 44px var(--accent-glow); transform: translateY(-2px); }
.btn-mint:disabled { cursor: progress; opacity: 0.85; }
.btn-mint.is-done { background: var(--pos); box-shadow: 0 10px 32px rgba(38, 208, 124, 0.4); }
.btn-mint .spinner {
width: 16px; height: 16px; border-radius: 50%;
border: 2px solid rgba(255, 255, 255, 0.4);
border-top-color: #fff;
animation: spin 0.7s linear infinite;
display: inline-block;
}
.mint-foot { font-size: 0.78rem; color: var(--muted); text-align: center; }
.addr {
color: var(--accent-2);
cursor: pointer;
border-bottom: 1px dashed rgba(63, 208, 255, 0.4);
}
.addr:hover { color: var(--text); }
/* ===== marquee ===== */
.marquee-section { overflow: hidden; padding: 1.5rem 0; mask: linear-gradient(90deg, transparent, #000 8%, #000 92%, transparent); }
.marquee { display: flex; gap: 1rem; width: max-content; animation: scroll 38s linear infinite; }
.marquee:hover { animation-play-state: paused; }
@keyframes scroll { to { transform: translateX(-50%); } }
.pfp {
width: 120px; height: 120px;
border-radius: var(--r-md);
flex: 0 0 auto;
border: 1px solid var(--line-2);
position: relative;
overflow: hidden;
}
/* ===== generic section ===== */
.section { padding: clamp(3rem, 8vw, 5.5rem) 0; }
.section-head { margin-bottom: 2.2rem; }
.eyebrow { font-family: var(--mono); font-size: 0.8rem; color: var(--accent); letter-spacing: 0.05em; }
.section-head h2 { font-size: clamp(1.8rem, 5vw, 2.8rem); margin-top: 0.5rem; }
.section-sub { color: var(--muted); margin-top: 0.6rem; }
/* lore */
.lore-grid { display: grid; gap: 2rem; }
.lore-lead { font-size: 1.2rem; max-width: 60ch; color: #d9cfffcc; }
.lore-lead b { color: var(--accent); }
.lore-cards { display: grid; grid-template-columns: repeat(3, 1fr); gap: 1rem; }
.lore-card {
background: var(--surface);
border: 1px solid var(--line);
border-radius: var(--r-lg);
padding: 1.5rem;
transition: border-color 0.2s, transform 0.2s;
}
.lore-card:hover { border-color: var(--accent-3); transform: translateY(-3px); }
.lore-ico { font-size: 1.6rem; color: var(--accent-2); filter: drop-shadow(0 0 10px var(--cyan-glow)); }
.lore-card h3 { font-size: 1.15rem; margin: 0.7rem 0 0.4rem; }
.lore-card p { color: var(--muted); font-size: 0.92rem; margin: 0; }
/* rarity */
.rarity-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(170px, 1fr)); gap: 1rem; }
.rarity-card {
position: relative;
border-radius: var(--r-lg);
overflow: hidden;
border: 1px solid var(--line);
background: var(--surface);
cursor: pointer;
transition: transform 0.25s, border-color 0.25s, box-shadow 0.25s;
}
.rarity-card:hover, .rarity-card:focus-visible {
transform: translateY(-5px);
border-color: var(--accent);
box-shadow: 0 16px 40px rgba(0, 0, 0, 0.5), 0 0 30px var(--accent-glow);
}
.rarity-art { aspect-ratio: 1; position: relative; }
.rarity-meta { padding: 0.8rem 0.9rem; display: flex; flex-direction: column; gap: 0.15rem; }
.rarity-name { font-weight: 600; font-size: 0.95rem; }
.rarity-tier {
font-family: var(--mono);
font-size: 0.72rem;
letter-spacing: 0.05em;
}
.tier-Legendary { color: var(--warn); }
.tier-Rare { color: var(--accent-2); }
.tier-Common { color: var(--muted); }
.rarity-overlay {
position: absolute;
inset: auto 0 0 0;
background: linear-gradient(180deg, transparent, rgba(13, 6, 24, 0.96));
padding: 0.9rem;
transform: translateY(101%);
transition: transform 0.28s cubic-bezier(0.22, 1, 0.36, 1);
font-size: 0.78rem;
}
.rarity-card:hover .rarity-overlay,
.rarity-card:focus-visible .rarity-overlay { transform: translateY(0); }
.rarity-overlay .trait-row { display: flex; justify-content: space-between; padding: 0.15rem 0; }
.rarity-overlay .trait-row span:last-child { color: var(--accent-2); font-family: var(--mono); }
/* timeline */
.timeline { list-style: none; margin: 0; padding: 0; position: relative; }
.timeline::before {
content: "";
position: absolute;
left: 9px; top: 6px; bottom: 6px;
width: 2px;
background: linear-gradient(180deg, var(--accent), var(--accent-2));
opacity: 0.4;
}
.tl-item { position: relative; padding: 0 0 1.8rem 2.4rem; }
.tl-dot {
position: absolute; left: 0; top: 4px;
width: 20px; height: 20px; border-radius: 50%;
background: var(--surface-2);
border: 2px solid var(--line-2);
}
.tl-item.done .tl-dot { background: var(--accent-2); border-color: var(--accent-2); box-shadow: 0 0 14px var(--cyan-glow); }
.tl-item.active .tl-dot { background: var(--accent); border-color: var(--accent); box-shadow: 0 0 16px var(--accent-glow); animation: pulse 1.6s infinite; }
.tl-phase { font-size: 0.75rem; letter-spacing: 0.08em; color: var(--muted); text-transform: uppercase; }
.tl-item.active .tl-phase { color: var(--accent); }
.tl-body h3 { font-size: 1.2rem; margin: 0.3rem 0; }
.tl-body p { color: var(--muted); margin: 0; font-size: 0.95rem; max-width: 56ch; }
/* team */
.team-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(180px, 1fr)); gap: 1rem; }
.team-card {
background: var(--surface);
border: 1px solid var(--line);
border-radius: var(--r-lg);
padding: 1.4rem;
text-align: center;
transition: transform 0.2s, border-color 0.2s;
}
.team-card:hover { transform: translateY(-4px); border-color: var(--accent-2); }
.team-ava {
width: 76px; height: 76px; border-radius: 50%;
margin: 0 auto 0.9rem;
border: 2px solid var(--line-2);
}
.team-name { font-weight: 600; }
.team-role { color: var(--accent-2); font-size: 0.82rem; }
.team-handle { font-family: var(--mono); font-size: 0.78rem; color: var(--muted); margin-top: 0.3rem; }
/* faq */
.faq { display: flex; flex-direction: column; gap: 0.7rem; }
.faq-item {
background: var(--surface);
border: 1px solid var(--line);
border-radius: var(--r-md);
overflow: hidden;
}
.faq-q {
width: 100%;
text-align: left;
background: transparent;
border: 0;
color: var(--text);
font-family: var(--font);
font-size: 1.02rem;
font-weight: 500;
padding: 1.1rem 1.2rem;
display: flex;
justify-content: space-between;
align-items: center;
gap: 1rem;
cursor: pointer;
}
.faq-q .chev { transition: transform 0.25s; color: var(--accent); font-size: 1.2rem; }
.faq-item.open .faq-q .chev { transform: rotate(45deg); }
.faq-item.open { border-color: var(--accent-3); }
.faq-a {
max-height: 0;
overflow: hidden;
transition: max-height 0.32s ease;
color: var(--muted);
}
.faq-a p { margin: 0; padding: 0 1.2rem 1.2rem; font-size: 0.95rem; }
/* community */
.community { padding: clamp(3rem, 8vw, 5rem) 0; }
.community-inner {
background: var(--grad);
border-radius: var(--r-lg);
padding: clamp(2rem, 6vw, 3.5rem);
text-align: center;
box-shadow: 0 24px 70px var(--accent-glow);
position: relative;
overflow: hidden;
}
.community-inner::after {
content: "";
position: absolute; inset: 0;
background: radial-gradient(60% 120% at 50% -20%, rgba(255, 255, 255, 0.25), transparent);
pointer-events: none;
}
.community-inner h2 { font-size: clamp(2rem, 6vw, 3.2rem); color: #fff; }
.community-inner p { color: rgba(255, 255, 255, 0.85); margin: 0.6rem 0 1.6rem; }
.community-btns { display: flex; gap: 0.8rem; justify-content: center; flex-wrap: wrap; position: relative; }
.community .btn-primary { background: #0d0618; box-shadow: 0 8px 24px rgba(0,0,0,0.4); }
.community .btn-ghost { background: rgba(255,255,255,0.16); border-color: rgba(255,255,255,0.4); color: #fff; }
/* footer */
.footer {
display: flex; justify-content: space-between; gap: 1rem; flex-wrap: wrap;
padding: 2rem 0 3rem;
font-size: 0.8rem;
border-top: 1px solid var(--line);
margin-top: 1rem;
}
/* toast */
.toast-host {
position: fixed;
bottom: 1.3rem; left: 50%;
transform: translateX(-50%);
z-index: 100;
display: flex; flex-direction: column; gap: 0.5rem;
align-items: center;
}
.toast {
background: var(--elevated);
border: 1px solid var(--line-2);
border-left: 3px solid var(--accent);
color: var(--text);
padding: 0.7rem 1.1rem;
border-radius: var(--r-md);
font-size: 0.9rem;
box-shadow: 0 12px 34px rgba(0, 0, 0, 0.5);
animation: toastIn 0.25s ease, toastOut 0.3s ease 2.6s forwards;
max-width: 90vw;
}
.toast.ok { border-left-color: var(--pos); }
@keyframes toastIn { from { opacity: 0; transform: translateY(12px); } }
@keyframes toastOut { to { opacity: 0; transform: translateY(12px); } }
/* ===== responsive ===== */
@media (max-width: 880px) {
.hero { grid-template-columns: 1fr; }
.lore-cards { grid-template-columns: 1fr; }
.nav-links { display: none; }
}
@media (max-width: 520px) {
.hero-stats { gap: 1.2rem; }
.stat-num { font-size: 1.4rem; }
.cd-cell b { font-size: 1.2rem; }
.mint-card-inner { padding: 1.2rem; }
.footer { flex-direction: column; gap: 0.4rem; }
.community-btns { flex-direction: column; }
.btn { width: 100%; }
.hero-cta .btn { width: auto; flex: 1; }
}
@media (prefers-reduced-motion: reduce) {
.aurora, .marquee, .brand-mark, .dot, .tl-item.active .tl-dot,
.glitch::before, .glitch::after { animation: none !important; }
}/* VOIDLINGS — NFT collection landing (UI simulation only). No wallet/RPC/on-chain calls. */
(function () {
"use strict";
/* ---------- config / mock state ---------- */
var TOTAL = 8888;
var PRICE = 0.069; // ◇ per mint
var MAX_PER_WALLET = 5;
var state = {
minted: 6213,
qty: 1,
connected: false,
minting: false,
};
var $ = function (id) { return document.getElementById(id); };
var nf = new Intl.NumberFormat("en-US");
/* ---------- toast ---------- */
function toast(msg, ok) {
var host = $("toastHost");
var el = document.createElement("div");
el.className = "toast" + (ok ? " ok" : "");
el.textContent = msg;
host.appendChild(el);
setTimeout(function () { el.remove(); }, 3000);
}
/* ---------- deterministic pseudo-random from seed ---------- */
function rng(seed) {
var s = seed % 2147483647;
if (s <= 0) s += 2147483646;
return function () { s = (s * 16807) % 2147483647; return (s - 1) / 2147483646; };
}
/* ---------- generative PFP (CSS-drawn) ---------- */
var BG_DUOS = [
["#ff4d8d", "#b86bff"], ["#3fd0ff", "#7c5cff"], ["#ff4d8d", "#3fd0ff"],
["#b86bff", "#3fd0ff"], ["#ff6ec7", "#ffb347"], ["#7c5cff", "#ff4d8d"],
];
function buildPfp(seed) {
var r = rng(seed);
var duo = BG_DUOS[Math.floor(r() * BG_DUOS.length)];
var angle = Math.floor(r() * 360);
var el = document.createElement("div");
el.style.background = "linear-gradient(" + angle + "deg," + duo[0] + "," + duo[1] + ")";
el.style.position = "relative";
el.style.width = "100%";
el.style.height = "100%";
// face plate
var face = document.createElement("div");
var fs = 44 + Math.floor(r() * 14);
face.style.cssText = "position:absolute;left:50%;top:52%;transform:translate(-50%,-50%);" +
"width:" + fs + "%;height:" + (fs + 6) + "%;border-radius:" + (18 + r() * 30) + "%;" +
"background:rgba(13,6,24,0.78);box-shadow:0 0 16px rgba(0,0,0,0.4) inset;";
el.appendChild(face);
// eyes
var eyeColors = ["#3fd0ff", "#ff4d8d", "#26d07c", "#ffb347", "#fff"];
var ec = eyeColors[Math.floor(r() * eyeColors.length)];
var ew = 9 + Math.floor(r() * 7);
function eye(left) {
var e = document.createElement("div");
e.style.cssText = "position:absolute;top:42%;" + (left ? "left:33%" : "right:33%") +
";width:" + ew + "%;aspect-ratio:1;border-radius:50%;background:" + ec +
";box-shadow:0 0 10px " + ec + ";transform:translateY(-50%);";
face.appendChild(e);
}
eye(true); eye(false);
// mouth glitch bar
var mouth = document.createElement("div");
mouth.style.cssText = "position:absolute;left:50%;bottom:22%;transform:translateX(-50%);" +
"width:" + (24 + r() * 22) + "%;height:" + (4 + r() * 5) + "%;border-radius:2px;background:" +
ec + ";opacity:0.85;";
face.appendChild(mouth);
// scanline glitch
var sl = document.createElement("div");
sl.style.cssText = "position:absolute;inset:0;background:repeating-linear-gradient(0deg," +
"rgba(255,255,255,0.05) 0 1px,transparent 1px 4px);mix-blend-mode:overlay;";
el.appendChild(sl);
return el;
}
/* ---------- marquee ---------- */
function buildMarquee() {
var wrap = $("marquee");
if (!wrap) return;
var n = 12;
for (var pass = 0; pass < 2; pass++) {
for (var i = 0; i < n; i++) {
var card = document.createElement("div");
card.className = "pfp";
card.appendChild(buildPfp(1000 + i * 37));
wrap.appendChild(card);
}
}
}
/* ---------- rarity grid ---------- */
var RARITY = [
{ name: "Void #0042", tier: "Legendary", traits: { Base: "Phantom", Eyes: "Singularity", Glitch: "Total", Score: "9.8" } },
{ name: "Void #1180", tier: "Rare", traits: { Base: "Plasma", Eyes: "Cyan Burn", Glitch: "Heavy", Score: "7.2" } },
{ name: "Void #3391", tier: "Common", traits: { Base: "Static", Eyes: "Dim", Glitch: "Light", Score: "3.1" } },
{ name: "Void #5567", tier: "Rare", traits: { Base: "Aurora", Eyes: "Pink Pulse", Glitch: "Heavy", Score: "6.9" } },
{ name: "Void #0777", tier: "Legendary", traits: { Base: "Genesis", Eyes: "Twin Sun", Glitch: "Corrupted", Score: "9.4" } },
{ name: "Void #6620", tier: "Common", traits: { Base: "Ash", Eyes: "Void", Glitch: "Faint", Score: "2.6" } },
{ name: "Void #2048", tier: "Rare", traits: { Base: "Neon", Eyes: "Strobe", Glitch: "Medium", Score: "5.8" } },
{ name: "Void #8888", tier: "Legendary", traits: { Base: "Omega", Eyes: "Eclipse", Glitch: "Maxed", Score: "10.0" } },
];
function buildRarity() {
var grid = $("rarityGrid");
if (!grid) return;
RARITY.forEach(function (item, idx) {
var card = document.createElement("article");
card.className = "rarity-card";
card.tabIndex = 0;
var art = document.createElement("div");
art.className = "rarity-art";
art.appendChild(buildPfp(7 + idx * 311));
var overlay = document.createElement("div");
overlay.className = "rarity-overlay";
Object.keys(item.traits).forEach(function (k) {
var row = document.createElement("div");
row.className = "trait-row";
row.innerHTML = "<span>" + k + "</span><span>" + item.traits[k] + "</span>";
overlay.appendChild(row);
});
art.appendChild(overlay);
var meta = document.createElement("div");
meta.className = "rarity-meta";
meta.innerHTML = '<span class="rarity-name">' + item.name + '</span>' +
'<span class="rarity-tier tier-' + item.tier + '">' + item.tier.toUpperCase() + '</span>';
card.appendChild(art);
card.appendChild(meta);
grid.appendChild(card);
});
}
/* ---------- team ---------- */
var TEAM = [
{ name: "0xNULL", role: "Founder / Static Architect", handle: "@0xnull" },
{ name: "Hex", role: "Generative Engine", handle: "@hex_renders" },
{ name: "Mira Vex", role: "Community / Swarm Ops", handle: "@miravex" },
{ name: "Glitchwizard", role: "Smart Contracts", handle: "@gwiz.eth" },
];
function buildTeam() {
var grid = $("teamGrid");
if (!grid) return;
TEAM.forEach(function (m, idx) {
var card = document.createElement("div");
card.className = "team-card";
var ava = document.createElement("div");
ava.className = "team-ava";
ava.appendChild(buildPfp(900 + idx * 53));
ava.firstChild.style.borderRadius = "50%";
card.appendChild(ava);
var info = document.createElement("div");
info.innerHTML = '<div class="team-name">' + m.name + '</div>' +
'<div class="team-role">' + m.role + '</div>' +
'<div class="team-handle">' + m.handle + '</div>';
card.appendChild(info);
grid.appendChild(card);
});
}
/* ---------- faq ---------- */
var FAQ = [
{ q: "What is a Voidling?", a: "A Voidling is a fully on-chain-seeded generative PFP minted on Lumen Chain. Each one is derived from your mint block hash, so no two glitches are alike." },
{ q: "How much does it cost to mint?", a: "Public mint is 0.069 ◇ per Voidling, with a maximum of 5 per wallet. Gas is paid in $LUMEN. This demo charges nothing — it is a UI simulation." },
{ q: "When is the reveal?", a: "Art reveals 24 hours after the public mint sells out, using an on-chain RNG seed so the order can be verified by anyone." },
{ q: "What utility do holders get?", a: "Voidnet access, $NOISE staking emissions, mutation labs, and gated seasonal void drops. Roadmap phases 02 and 03 unlock these." },
{ q: "Is the art CC0?", a: "Yes. Remix, print, animate, or fork your Voidling anywhere. The static belongs to the swarm." },
];
function buildFaq() {
var list = $("faq-list");
if (!list) return;
FAQ.forEach(function (item) {
var wrap = document.createElement("div");
wrap.className = "faq-item";
var btn = document.createElement("button");
btn.className = "faq-q";
btn.type = "button";
btn.setAttribute("aria-expanded", "false");
btn.innerHTML = '<span>' + item.q + '</span><span class="chev" aria-hidden="true">+</span>';
var ans = document.createElement("div");
ans.className = "faq-a";
ans.innerHTML = "<p>" + item.a + "</p>";
btn.addEventListener("click", function () {
var open = wrap.classList.toggle("open");
btn.setAttribute("aria-expanded", open ? "true" : "false");
ans.style.maxHeight = open ? ans.scrollHeight + "px" : "0";
});
wrap.appendChild(btn);
wrap.appendChild(ans);
list.appendChild(wrap);
});
}
/* ---------- animated count-up (hero stats) ---------- */
function animateCounts() {
document.querySelectorAll(".stat-num[data-count]").forEach(function (el) {
var target = parseFloat(el.getAttribute("data-count"));
var dec = parseInt(el.getAttribute("data-dec") || "0", 10);
var start = performance.now();
var dur = 1400;
function tick(now) {
var t = Math.min(1, (now - start) / dur);
var eased = 1 - Math.pow(1 - t, 3);
var val = target * eased;
el.textContent = dec ? val.toFixed(dec) : nf.format(Math.round(val));
if (t < 1) requestAnimationFrame(tick);
}
requestAnimationFrame(tick);
});
}
/* ---------- progress ---------- */
function renderProgress() {
var pct = (state.minted / TOTAL) * 100;
$("progressFill").style.width = pct.toFixed(2) + "%";
$("mintedLabel").textContent = nf.format(state.minted) + " / " + nf.format(TOTAL) + " minted";
$("pctLabel").textContent = pct.toFixed(1) + "%";
var bar = $("progressBar");
bar.setAttribute("aria-valuenow", String(state.minted));
}
/* ---------- qty + cost ---------- */
function renderCost() {
$("qtyVal").textContent = String(state.qty);
$("costVal").textContent = (state.qty * PRICE).toFixed(3) + " ◇";
$("qtyMinus").disabled = state.qty <= 1 || state.minting;
$("qtyPlus").disabled = state.qty >= MAX_PER_WALLET || state.minting;
}
/* ---------- countdown ---------- */
function startCountdown() {
var end = Date.now() + (2 * 24 * 3600 + 7 * 3600 + 41 * 60 + 12) * 1000;
function pad(n) { return String(n).padStart(2, "0"); }
function tick() {
var diff = Math.max(0, end - Date.now());
var s = Math.floor(diff / 1000);
$("cd-d").textContent = pad(Math.floor(s / 86400));
$("cd-h").textContent = pad(Math.floor((s % 86400) / 3600));
$("cd-m").textContent = pad(Math.floor((s % 3600) / 60));
$("cd-s").textContent = pad(s % 60);
}
tick();
setInterval(tick, 1000);
}
/* ---------- mint flow (simulated) ---------- */
function doMint() {
if (state.minting) return;
if (!state.connected) {
toast("Connect your wallet to mint.");
return;
}
if (state.minted + state.qty > TOTAL) {
toast("Not enough supply left for that quantity.");
return;
}
state.minting = true;
var btn = $("mintBtn");
var label = btn.querySelector(".mint-btn-label");
btn.disabled = true;
label.innerHTML = '<span class="spinner"></span> Confirm in wallet…';
renderCost();
// simulated signing + confirmation
setTimeout(function () {
label.textContent = "Minting on Lumen Chain…";
}, 900);
setTimeout(function () {
var n = state.qty;
state.minted += n;
renderProgress();
state.minting = false;
btn.classList.add("is-done");
label.textContent = "✓ Minted " + n + " Voidling" + (n > 1 ? "s" : "") + "!";
toast("Success — minted " + n + " Voidling" + (n > 1 ? "s" : "") + ". Reveal in 24h.", true);
setTimeout(function () {
btn.classList.remove("is-done");
btn.disabled = false;
label.textContent = "Mint now";
renderCost();
}, 2600);
}, 2100);
}
/* ---------- wallet connect (simulated) ---------- */
function connectWallet() {
if (state.connected) {
toast("Wallet already connected: 0x4f2a…9e1b");
return;
}
var btn = $("connectBtn");
btn.textContent = "Connecting…";
setTimeout(function () {
state.connected = true;
btn.textContent = "0x4f2a…9e1b";
btn.classList.add("connected");
toast("Wallet connected on Lumen Chain.", true);
}, 800);
}
/* ---------- wire up ---------- */
function init() {
buildMarquee();
buildRarity();
buildTeam();
buildFaq();
renderProgress();
renderCost();
startCountdown();
animateCounts();
$("qtyMinus").addEventListener("click", function () {
if (state.qty > 1) { state.qty--; renderCost(); }
});
$("qtyPlus").addEventListener("click", function () {
if (state.qty < MAX_PER_WALLET) { state.qty++; renderCost(); }
else toast("Max " + MAX_PER_WALLET + " per wallet.");
});
$("mintBtn").addEventListener("click", doMint);
$("connectBtn").addEventListener("click", connectWallet);
var addr = $("contractAddr");
function copyAddr() {
var full = "0x7a3f4b2c9d8e1f0a6b5c4d3e2f1a0b9c8d7e6f5a";
if (navigator.clipboard) navigator.clipboard.writeText(full).catch(function () {});
toast("Contract address copied.", true);
}
addr.addEventListener("click", copyAddr);
addr.addEventListener("keydown", function (e) {
if (e.key === "Enter" || e.key === " ") { e.preventDefault(); copyAddr(); }
});
document.querySelectorAll("[data-social]").forEach(function (b) {
b.addEventListener("click", function () {
toast("Opening " + b.getAttribute("data-social") + " (demo — no real link).");
});
});
}
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", init);
} else {
init();
}
})();<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>VOIDLINGS — Genesis Mint</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=Space+Grotesk:wght@400;500;600;700&family=JetBrains+Mono:wght@400;500;700&display=swap" rel="stylesheet" />
<link rel="stylesheet" href="style.css" />
</head>
<body>
<div class="aurora" aria-hidden="true"></div>
<!-- ============ NAV ============ -->
<header class="nav">
<a class="brand" href="#top">
<span class="brand-mark" aria-hidden="true"></span>
<span class="brand-name">VOIDLINGS</span>
</a>
<nav class="nav-links" aria-label="Primary">
<a href="#about">Lore</a>
<a href="#rarity">Traits</a>
<a href="#roadmap">Roadmap</a>
<a href="#team">Team</a>
<a href="#faq">FAQ</a>
</nav>
<button class="btn btn-ghost connect" id="connectBtn" type="button">Connect Wallet</button>
</header>
<!-- ============ HERO ============ -->
<main id="top">
<section class="hero">
<div class="hero-copy">
<span class="pill">
<span class="dot" aria-hidden="true"></span>
Live on Lumen Chain
</span>
<h1 class="hero-title">
<span class="glitch" data-text="VOIDLINGS">VOIDLINGS</span>
</h1>
<p class="hero-sub">
8,888 hand-corrupted PFPs born from the static between blocks.
Summon yours, join the swarm, and ascend the void.
</p>
<div class="hero-stats">
<div class="stat">
<span class="stat-num" data-count="8888">0</span>
<span class="stat-label">Supply</span>
</div>
<div class="stat">
<span class="stat-num" data-count="0.069" data-dec="3">0</span>
<span class="stat-label">Mint price ◇</span>
</div>
<div class="stat">
<span class="stat-num" data-count="12420">0</span>
<span class="stat-label">Holders</span>
</div>
</div>
<div class="hero-cta">
<a class="btn btn-primary" href="#mint">Mint now</a>
<a class="btn btn-ghost" href="#about">Read the lore</a>
</div>
</div>
<!-- mint panel -->
<aside class="mint-card" id="mint" aria-label="Mint panel">
<div class="mint-card-inner">
<div class="mint-head">
<div>
<h2 class="mint-title">Genesis Mint</h2>
<p class="mint-sub">Public phase · max 5 per wallet</p>
</div>
<span class="badge-live">
<span class="dot" aria-hidden="true"></span> LIVE
</span>
</div>
<!-- countdown -->
<div class="countdown" aria-live="polite">
<span class="cd-label">Public mint ends in</span>
<div class="cd-grid" id="countdown">
<div class="cd-cell"><b id="cd-d">00</b><span>days</span></div>
<div class="cd-cell"><b id="cd-h">00</b><span>hrs</span></div>
<div class="cd-cell"><b id="cd-m">00</b><span>min</span></div>
<div class="cd-cell"><b id="cd-s">00</b><span>sec</span></div>
</div>
</div>
<!-- progress -->
<div class="progress-wrap">
<div class="progress-meta">
<span id="mintedLabel" class="mono">0 / 8,888 minted</span>
<span id="pctLabel" class="mono">0%</span>
</div>
<div class="progress" role="progressbar" aria-valuemin="0" aria-valuemax="8888" aria-valuenow="0" id="progressBar">
<div class="progress-fill" id="progressFill"></div>
</div>
</div>
<!-- quantity -->
<div class="qty-row">
<span class="qty-label">Quantity</span>
<div class="stepper" role="group" aria-label="Mint quantity">
<button class="step-btn" type="button" id="qtyMinus" aria-label="Decrease quantity">–</button>
<span class="qty-val mono" id="qtyVal" aria-live="polite">1</span>
<button class="step-btn" type="button" id="qtyPlus" aria-label="Increase quantity">+</button>
</div>
</div>
<div class="cost-row">
<span class="muted">Total</span>
<span class="cost mono" id="costVal">0.069 ◇</span>
</div>
<button class="btn btn-mint" id="mintBtn" type="button">
<span class="mint-btn-label">Mint now</span>
</button>
<p class="mint-foot mono">
Contract <span class="addr" id="contractAddr" title="Copy contract address" role="button" tabindex="0">0x7a3f…c41d</span> · Lumen Chain
</p>
</div>
</aside>
</section>
<!-- ============ MARQUEE ============ -->
<section class="marquee-section" aria-label="Collection preview">
<div class="marquee" id="marquee" aria-hidden="true"></div>
</section>
<!-- ============ ABOUT / LORE ============ -->
<section class="section" id="about">
<div class="section-head">
<span class="eyebrow">// 001 — origin</span>
<h2>Born from the static</h2>
</div>
<div class="lore-grid">
<p class="lore-lead">
When Lumen Chain forked at block 0, the discarded entropy didn't vanish —
it congealed. The <b>Voidlings</b> are what crawled out of that leftover noise:
8,888 sentient glitches, each one a unique corruption of the genesis seed.
</p>
<div class="lore-cards">
<article class="lore-card">
<span class="lore-ico" aria-hidden="true">◈</span>
<h3>On-chain seeded</h3>
<p>Every trait is derived deterministically from your mint block hash. Provably weird.</p>
</article>
<article class="lore-card">
<span class="lore-ico" aria-hidden="true">⚡</span>
<h3>Swarm utility</h3>
<p>Holders gate the Voidnet — staking, mutation labs, and seasonal void drops.</p>
</article>
<article class="lore-card">
<span class="lore-ico" aria-hidden="true">✦</span>
<h3>CC0 spirit</h3>
<p>Remix your Voidling anywhere. The static belongs to everyone who escaped it.</p>
</article>
</div>
</div>
</section>
<!-- ============ RARITY / TRAITS ============ -->
<section class="section" id="rarity">
<div class="section-head">
<span class="eyebrow">// 002 — traits</span>
<h2>Rarity & traits</h2>
<p class="section-sub">Hover a specimen to read its corruption profile.</p>
</div>
<div class="rarity-grid" id="rarityGrid"></div>
</section>
<!-- ============ ROADMAP ============ -->
<section class="section" id="roadmap">
<div class="section-head">
<span class="eyebrow">// 003 — trajectory</span>
<h2>The descent</h2>
</div>
<ol class="timeline">
<li class="tl-item done">
<span class="tl-dot" aria-hidden="true"></span>
<div class="tl-body">
<span class="tl-phase mono">PHASE 00 · complete</span>
<h3>Genesis static</h3>
<p>Contract deployed, allowlist snapshot, generative engine seeded on Lumen Chain.</p>
</div>
</li>
<li class="tl-item active">
<span class="tl-dot" aria-hidden="true"></span>
<div class="tl-body">
<span class="tl-phase mono">PHASE 01 · now</span>
<h3>Public summon</h3>
<p>8,888 Voidlings open to the swarm. Reveal 24h post-mint via on-chain RNG.</p>
</div>
</li>
<li class="tl-item">
<span class="tl-dot" aria-hidden="true"></span>
<div class="tl-body">
<span class="tl-phase mono">PHASE 02 · Q3</span>
<h3>Mutation labs</h3>
<p>Burn two Voidlings to splice traits into a 1/1 Aberration. Risky. Glorious.</p>
</div>
</li>
<li class="tl-item">
<span class="tl-dot" aria-hidden="true"></span>
<div class="tl-body">
<span class="tl-phase mono">PHASE 03 · Q4</span>
<h3>The Voidnet</h3>
<p>Token-gated world, staking emissions ($NOISE), and the first seasonal void drop.</p>
</div>
</li>
</ol>
</section>
<!-- ============ TEAM ============ -->
<section class="section" id="team">
<div class="section-head">
<span class="eyebrow">// 004 — the architects</span>
<h2>The swarm core</h2>
</div>
<div class="team-grid" id="teamGrid"></div>
</section>
<!-- ============ FAQ ============ -->
<section class="section" id="faq">
<div class="section-head">
<span class="eyebrow">// 005 — questions</span>
<h2>FAQ</h2>
</div>
<div class="faq" id="faq-list"></div>
</section>
<!-- ============ COMMUNITY CTA ============ -->
<section class="community" aria-label="Join the community">
<div class="community-inner">
<h2>Join the swarm</h2>
<p>40k+ glitches and counting. The void is louder together.</p>
<div class="community-btns">
<button class="btn btn-primary" type="button" data-social="Discord">
<span aria-hidden="true">◇</span> Discord
</button>
<button class="btn btn-ghost" type="button" data-social="X">
<span aria-hidden="true">𝕏</span> Follow on X
</button>
</div>
</div>
</section>
</main>
<footer class="footer">
<span class="mono">VOIDLINGS · Lumen Chain · CC0</span>
<span class="mono muted">UI simulation — no real wallet or on-chain calls.</span>
</footer>
<div class="toast-host" id="toastHost" aria-live="polite" aria-atomic="true"></div>
<script src="script.js"></script>
</body>
</html>NFT Project / Collection Landing
A bold, glow-soaked landing page for VOIDLINGS, a fictional 8,888-piece generative PFP collection on the made-up Lumen Chain. The hero pairs a glitching gradient wordmark and count-up stats with a glassy mint panel: a live countdown, a minted-versus-total progress bar, a quantity stepper capped per wallet, an animated total cost, and a “Mint now” button.
Every avatar in the page is drawn entirely in CSS/JS from a deterministic seed — no images. A paused-on-hover marquee scrolls these generative thumbnails, while the rarity section reveals each specimen’s trait profile on hover or focus. The mint button runs a simulated flow with “confirm in wallet”, “minting”, and a success state that advances the progress bar; the connect button and contract address show realistic, clearly fictional feedback via a toast helper.
Supporting sections cover the project lore, a roadmap timeline, a team grid, and an accordion FAQ, finished with a Discord/X community CTA. The whole page is responsive down to ~360px, keyboard-usable, and respects reduced-motion preferences.
UI-only simulation — no real wallet, RPC, or on-chain calls. Mock data, fictional tokens.