File size: 5,746 Bytes
cd5dc96
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Odometer Counter — Cinematic Module</title>
<link href="https://fonts.googleapis.com/css2?family=Outfit:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500;700&display=swap" rel="stylesheet">
<script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/3.12.5/gsap.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/3.12.5/ScrollTrigger.min.js"></script>
<style>

  :root{--bg:#0a0a0b;--text:#eae7e2;--muted:#5a5a5e;--accent:#5eadb5;--border:#1e1e22}

  *{margin:0;padding:0;box-sizing:border-box}body{background:var(--bg);color:var(--text);font-family:'Outfit',sans-serif;-webkit-font-smoothing:antialiased;overflow-x:hidden}

  .hero{min-height:100dvh;display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;padding:40px 24px}

  .hero-label{font-size:12px;letter-spacing:.15em;text-transform:uppercase;color:var(--muted);margin-bottom:20px}

  .hero h1{font-size:clamp(32px,6vw,64px);font-weight:600;line-height:1.15;max-width:700px;letter-spacing:-.025em}

  .hero h1 span{color:var(--accent)}

  .stats-section{padding:80px 24px;max-width:900px;margin:0 auto}

  .stats-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(200px,1fr));gap:32px;text-align:center}

  .stat-item{padding:40px 20px;background:#111114;border:1px solid var(--border);border-radius:20px}

  .odometer{font-family:'JetBrains Mono',monospace;font-size:clamp(36px,6vw,64px);font-weight:700;color:var(--accent);letter-spacing:-.03em;overflow:hidden;height:1.15em;display:flex;justify-content:center}

  .odo-digit{display:inline-block;overflow:hidden;height:1.15em;position:relative}

  .odo-strip{display:flex;flex-direction:column;transition:transform 1.5s cubic-bezier(.16,1,.3,1)}

  .odo-strip span{display:block;height:1.15em;line-height:1.15}

  .stat-suffix{font-family:'JetBrains Mono',monospace;font-size:clamp(24px,4vw,40px);font-weight:700;color:var(--accent);vertical-align:top}

  .stat-label{font-size:14px;color:var(--muted);margin-top:8px}

  .explain{padding:120px 24px;text-align:center;max-width:680px;margin:0 auto;border-top:1px solid #1a1a1d}

  .explain .tag{display:inline-block;font-size:11px;letter-spacing:.12em;text-transform:uppercase;color:var(--accent);border:1px solid rgba(94,173,181,.2);padding:4px 12px;border-radius:100px;margin-bottom:20px}

  .explain h2{font-size:clamp(24px,4vw,40px);font-weight:600;letter-spacing:-.02em;margin-bottom:16px}

  .explain p{font-size:17px;color:var(--muted);line-height:1.6}

  footer{border-top:1px solid #1a1a1d;padding:24px;text-align:center;font-size:12px;color:var(--muted)}

</style>
</head>
<body>
<section class="hero">
  <div class="hero-label">Module 20 — Odometer Counter</div>
  <h1>Numbers <span>roll like wheels</span> into place</h1>
  <p>Mechanical odometer-style digit strips that spin to the target number on scroll. More physical than a simple count-up.</p>
</section>
<section class="stats-section">
  <div class="stats-grid">
    <div class="stat-item"><div class="odometer" data-value="2847"></div><div class="stat-label">Projects delivered</div></div>
    <div class="stat-item"><div class="odometer" data-value="99" data-suffix="%"></div><div class="stat-label">Client satisfaction</div></div>
    <div class="stat-item"><div class="odometer" data-value="14"></div><div class="stat-label">Global offices</div></div>
    <div class="stat-item"><div class="odometer" data-value="340" data-suffix="ms"></div><div class="stat-label">Avg response time</div></div>
  </div>
</section>
<section class="explain">
  <div class="tag">Why it works</div>
  <h2>Mechanical trust</h2>
  <p>The rolling digits feel like a physical counter incrementing. It's more satisfying than a simple number appearing — the motion implies the number was earned, not stated. Stats sections, dashboards, and pricing pages all benefit.</p>
</section>
<footer>Cinematic Module — Odometer Counter</footer>
<script>

(function(){

  gsap.registerPlugin(ScrollTrigger);

  document.querySelectorAll('.odometer').forEach(function(odo){

    var val=odo.dataset.value;var suffix=odo.dataset.suffix||'';

    var digits=val.split('');

    odo.innerHTML='';

    digits.forEach(function(d){

      var digit=document.createElement('div');digit.className='odo-digit';

      var strip=document.createElement('div');strip.className='odo-strip';

      for(var i=0;i<=9;i++){var s=document.createElement('span');s.textContent=i;strip.appendChild(s)}

      digit.appendChild(strip);odo.appendChild(digit);

    });

    if(suffix){var s=document.createElement('span');s.className='stat-suffix';s.textContent=suffix;odo.appendChild(s)}

    ScrollTrigger.create({trigger:odo,start:'top 85%',once:true,onEnter:function(){

      var strips=odo.querySelectorAll('.odo-strip');

      strips.forEach(function(strip,i){

        var target=parseInt(digits[i]);

        var h=strip.children[0].offsetHeight;

        strip.style.transform='translateY(-'+(target*h)+'px)';

        strip.style.transitionDelay=(i*0.12)+'s';

      });

    }});

  });

})();

</script>
<div style="position:fixed;bottom:0;left:0;right:0;padding:8px 16px;font-size:11px;color:#5a5a5e;text-align:center;z-index:99;background:rgba(9,9,11,.85);backdrop-filter:blur(8px);border-top:1px solid #1e1e22;font-family:Outfit,sans-serif">&copy; Created by Jay from <a href="https://robonuggets.com" style="color:#e8793a;text-decoration:none">RoboLabs</a>. Learn more at <a href="https://robonuggets.com" style="color:#e8793a;text-decoration:none">RoboNuggets</a></div>
</body>
</html>