{ CF }

12 CSS Scroll Animations

Scroll Reveal Suite

An architecture-studio portfolio showcasing six reveal techniques: mask-slide, lift, scale-curtain, clip, wipe, and blur dissolve.

CSS + JS MIT licensed

Scroll Reveal Suite the 7th of 12 designs in the 12 CSS Scroll Animations collection. The design pairs CSS styling with a small amount of JavaScript for interactivity. Copy the HTML, CSS and JavaScript panels below into your project — the JS is self-contained, has zero dependencies, and is safe to drop into any framework (React, Vue, Svelte, plain HTML). The design honours prefers-reduced-motion and uses real semantic markup, so it ships accessibility-ready out of the box.

Live preview

This is a full-page demo — interact inside the frame above, or open it in the playground for the full-screen experience.

Open in playground

The code

<!-- HERO: mask slide reveal -->
<section class="hero">
  <div class="hero-eyebrow" data-r="lift">Architectural Practice · Est. 2009</div>
  <div class="hero-wordmark">
    <div class="m-wrap"><div class="m-inner">MURANO</div></div>
    <div class="m-wrap"><div class="m-inner" style="transition-delay:0.08s">STUDIO</div></div>
  </div>
  <div class="hero-sub">
    <p class="hero-desc" data-r="lift" data-d="2">We design spaces that endure — buildings that carry their own silence, interiors that remember why they were made.</p>
    <div class="hero-year" data-r="dissolve" data-d="3">2024</div>
  </div>
</section>

<hr class="rule">

<!-- APPROACH: line-by-line mask reveal -->
<section>
  <div class="approach-label" data-r="lift">Our philosophy</div>
  <p class="approach-text">
    <span class="line"><span class="m-wrap"><span class="m-inner">Architecture is not about buildings.</span></span></span>
    <span class="line"><span class="m-wrap"><span class="m-inner">It is about <em>what buildings make possible</em> —</span></span></span>
    <span class="line"><span class="m-wrap"><span class="m-inner" style="transition-delay:0.1s">the meetings, the silences, the accidental</span></span></span>
    <span class="line"><span class="m-wrap"><span class="m-inner" style="transition-delay:0.18s">encounters that shape a life.</span></span></span>
  </p>
</section>

<hr class="rule">

<!-- PROJECTS: curtain + wipe reveals -->
<section>
  <div class="projects-head">
    <h2 class="m-wrap"><span class="m-inner">Selected Work</span></h2>
    <span data-r="lift">2019 — 2024</span>
  </div>
  <div class="project-grid">
    <div class="project-card">
      <div class="proj-num" data-r="lift">01</div>
      <div class="proj-panel" data-r="curtain"><div class="proj-img pi1"></div></div>
      <div class="proj-name" data-r="lift" data-d="1">Meridian House</div>
      <div class="proj-meta" data-r="lift" data-d="2">Residential · Copenhagen, 2022</div>
    </div>
    <div class="project-card">
      <div class="proj-num" data-r="lift" data-d="1">02</div>
      <div class="proj-panel" data-r="curtain" data-d="1"><div class="proj-img pi2"></div></div>
      <div class="proj-name" data-r="lift" data-d="2">Tidal Pavilion</div>
      <div class="proj-meta" data-r="lift" data-d="3">Cultural · Oslo, 2023</div>
    </div>
    <div class="project-card">
      <div class="proj-num" data-r="lift" data-d="2">03</div>
      <div class="proj-panel" data-r="curtain" data-d="2"><div class="proj-img pi3"></div></div>
      <div class="proj-name" data-r="lift" data-d="3">Thermal Baths</div>
      <div class="proj-meta" data-r="lift" data-d="4">Public · Reykjavik, 2021</div>
    </div>
    <div class="project-card">
      <div class="proj-num" data-r="lift" data-d="3">04</div>
      <div class="proj-panel" data-r="curtain" data-d="3"><div class="proj-img pi4"></div></div>
      <div class="proj-name" data-r="lift" data-d="4">Canopy Hall</div>
      <div class="proj-meta" data-r="lift" data-d="5">Education · Amsterdam, 2024</div>
    </div>
  </div>
</section>

<!-- NUMBERS: scale reveal -->
<section class="numbers">
  <div class="numbers-grid">
    <div class="num-cell"><div class="num-val" data-r="scale">48</div><div class="num-lbl">Projects built</div></div>
    <div class="num-cell" data-d="1"><div class="num-val" data-r="scale" data-d="1">14</div><div class="num-lbl">Countries</div></div>
    <div class="num-cell" data-d="2"><div class="num-val" data-r="scale" data-d="2">9</div><div class="num-lbl">Awards won</div></div>
    <div class="num-cell" data-d="3"><div class="num-val" data-r="scale" data-d="3">15yr</div><div class="num-lbl">In practice</div></div>
  </div>
</section>

<!-- QUOTE: wipe reveal -->
<section class="quote-section">
  <div class="quote-rule" data-r="wipe"></div>
  <blockquote class="quote-text" data-r="curtain"><span>"Good architecture makes you forget the building. You only remember what happened inside it."</span></blockquote>
  <p class="quote-attr" data-r="lift" data-d="2">— Founder, Murano Studio</p>
</section>

<!-- CLOSE: dissolve reveal -->
<section class="close-section">
  <div class="cta-label" data-r="lift">Begin a project</div>
  <div class="cta-text">
    <div class="m-wrap"><div class="m-inner">LET'S BUILD</div></div>
    <div class="m-wrap"><div class="m-inner" style="transition-delay:0.08s"><em style="font-family:'DM Serif Display',serif;font-style:italic;color:var(--gold)">something</em></div></div>
    <div class="m-wrap"><div class="m-inner" style="transition-delay:0.16s">LASTING</div></div>
  </div>
  <a href="#" class="cta-link" data-r="lift" data-d="3">[email protected] &nbsp;→</a>
</section>
*,*::before,*::after{box-sizing:border-box;margin:0;padding:0}
:root{--ink:#0A0806;--paper:#F4F0E8;--gold:#C8A252;--smoke:rgba(200,162,82,0.12)}
body{background:var(--ink);color:var(--paper);font-family:'Outfit',sans-serif;overflow-x:hidden}

/* REVEAL PRIMITIVES */

/* 1: mask slide — text rises from below a hidden overflow */
.m-wrap{overflow:hidden;display:block}
.m-inner{
  display:block;
  transform:translateY(108%);
  transition:transform 1.05s cubic-bezier(0.16,1,0.3,1);
}
.m-inner.in{transform:translateY(0)}

/* 2: fade + lift — classic, refined */
[data-r="lift"]{
  opacity:0;transform:translateY(36px);
  transition:opacity 0.9s ease,transform 0.9s cubic-bezier(0.16,1,0.3,1);
}
[data-r="lift"].in{opacity:1;transform:translateY(0)}

/* 3: scale reveal — panel expands from compressed */
[data-r="scale"]{
  opacity:0;transform:scaleY(0.6) translateY(20px);transform-origin:top center;
  transition:opacity 0.85s ease,transform 0.85s cubic-bezier(0.16,1,0.3,1);
}
[data-r="scale"].in{opacity:1;transform:scaleY(1) translateY(0)}

/* 4: clip from bottom — curtain rises.
   The clip lives on the INNER child, not the [data-r] element itself:
   a fully clip-path-clipped element has zero visible area, so an
   IntersectionObserver watching it would report ratio 0 forever and
   the reveal could never fire (chicken-and-egg). The observed
   [data-r="curtain"] element stays unclipped (just overflow:hidden);
   its child carries the curtain. */
[data-r="curtain"]{overflow:hidden}
[data-r="curtain"]>*{
  display:block;
  clip-path:inset(0 0 100% 0);
  transition:clip-path 1.1s cubic-bezier(0.16,1,0.3,1);
}
[data-r="curtain"].in>*{clip-path:inset(0 0 0% 0)}

/* 5: horizontal wipe */
[data-r="wipe"]{
  clip-path:inset(0 100% 0 0);
  transition:clip-path 1s cubic-bezier(0.16,1,0.3,1);
}
[data-r="wipe"].in{clip-path:inset(0 0% 0 0)}

/* 6: blur dissolve */
[data-r="dissolve"]{
  opacity:0;filter:blur(18px);transform:scale(1.04);
  transition:opacity 1.1s ease,filter 1s ease,transform 1.1s ease;
}
[data-r="dissolve"].in{opacity:1;filter:blur(0);transform:scale(1)}

/* stagger helpers */
[data-d="1"]{transition-delay:0.08s}
[data-d="2"]{transition-delay:0.16s}
[data-d="3"]{transition-delay:0.24s}
[data-d="4"]{transition-delay:0.32s}
[data-d="5"]{transition-delay:0.4s}
[data-d="6"]{transition-delay:0.48s}

/* LAYOUT */
section{padding:8rem 6vw;position:relative}
hr.rule{border:none;border-top:1px solid rgba(244,240,232,0.1);margin:0 6vw}

/* HERO */
.hero{
  min-height:100vh;display:flex;flex-direction:column;
  justify-content:flex-end;padding-bottom:6rem;
  border-bottom:1px solid rgba(244,240,232,0.08);
}
.hero-eyebrow{
  font-size:11px;letter-spacing:0.22em;text-transform:uppercase;
  color:var(--gold);margin-bottom:3rem;
  display:flex;align-items:center;gap:1rem;
}
.hero-eyebrow::before{content:'';width:40px;height:1px;background:var(--gold)}
.hero-wordmark{
  font-family:'Bebas Neue',sans-serif;
  font-size:clamp(80px,14vw,200px);
  line-height:0.88;letter-spacing:-0.01em;
  color:var(--paper);
}
.hero-sub{
  margin-top:3rem;display:flex;justify-content:space-between;align-items:flex-end;
}
.hero-desc{
  font-size:clamp(14px,1.4vw,18px);font-weight:300;
  color:rgba(244,240,232,0.5);max-width:400px;line-height:1.7;
}
.hero-year{
  font-family:'DM Serif Display',serif;font-style:italic;
  font-size:clamp(60px,8vw,120px);color:transparent;
  -webkit-text-stroke:1px rgba(244,240,232,0.12);
}

/* APPROACH */
.approach-label{
  font-size:10px;letter-spacing:0.22em;text-transform:uppercase;
  color:var(--gold);margin-bottom:5rem;
}
.approach-text{
  font-family:'DM Serif Display',serif;
  font-size:clamp(28px,3.8vw,60px);line-height:1.25;
  max-width:860px;
}
.approach-text em{font-style:italic;color:var(--gold)}
.approach-text .line{overflow:hidden;display:block;padding-bottom:0.05em}

/* PROJECTS */
.projects-head{
  display:flex;justify-content:space-between;align-items:baseline;
  margin-bottom:4rem;
}
.projects-head h2{
  font-family:'Bebas Neue',sans-serif;
  font-size:clamp(36px,5vw,72px);letter-spacing:0.02em;
}
.projects-head span{
  font-size:11px;letter-spacing:0.18em;text-transform:uppercase;
  color:rgba(244,240,232,0.3);
}
.project-grid{display:grid;grid-template-columns:1fr 1fr;gap:1px;background:rgba(244,240,232,0.08)}
.project-card{
  background:var(--ink);padding:3rem 2.5rem;
  border:none;position:relative;overflow:hidden;
}
.project-card::after{
  content:'';position:absolute;inset:0;
  background:var(--smoke);opacity:0;
  transition:opacity 0.4s ease;pointer-events:none;
}
.project-card:hover::after{opacity:1}
.proj-num{
  font-family:'DM Serif Display',serif;font-size:11px;
  color:var(--gold);letter-spacing:0.1em;margin-bottom:2rem;
}
.proj-panel{
  height:240px;margin-bottom:2rem;border-radius:4px;overflow:hidden;
  position:relative;
}
.proj-img{width:100%;height:100%;position:relative}
.pi1{background:linear-gradient(145deg,#1A1408 0%,#2E240E 40%,#3D320F 100%)}
.pi2{background:linear-gradient(145deg,#0A0E14 0%,#141E2A 40%,#1E2E40 100%)}
.pi3{background:linear-gradient(145deg,#100A0A 0%,#201010 40%,#2A1212 100%)}
.pi4{background:linear-gradient(145deg,#0A100A 0%,#121C10 40%,#1A2A14 100%)}
.proj-name{
  font-family:'Bebas Neue',sans-serif;font-size:clamp(28px,3vw,44px);
  letter-spacing:0.02em;margin-bottom:0.5rem;
}
.proj-meta{font-size:12px;color:rgba(244,240,232,0.3);letter-spacing:0.06em}

/* NUMBERS */
.numbers{background:#0E0C09;border-top:1px solid rgba(244,240,232,0.06)}
.numbers-grid{display:grid;grid-template-columns:repeat(4,1fr);gap:0;border:1px solid rgba(244,240,232,0.06)}
.num-cell{padding:3rem 2rem;border-right:1px solid rgba(244,240,232,0.06)}
.num-cell:last-child{border-right:none}
.num-val{
  font-family:'Bebas Neue',sans-serif;
  font-size:clamp(48px,5vw,80px);line-height:1;
  color:var(--gold);margin-bottom:0.5rem;
}
.num-lbl{font-size:11px;letter-spacing:0.14em;text-transform:uppercase;color:rgba(244,240,232,0.3)}

/* MANIFESTO QUOTE */
.quote-section{background:#0A0906}
.quote-rule{width:60px;height:1px;background:var(--gold);margin-bottom:4rem}
.quote-text{
  font-family:'DM Serif Display',serif;font-style:italic;
  font-size:clamp(28px,4vw,64px);line-height:1.2;max-width:900px;
}
.quote-attr{
  margin-top:3rem;font-size:12px;letter-spacing:0.16em;text-transform:uppercase;
  color:rgba(244,240,232,0.3);
}

/* CLOSE */
.close-section{
  min-height:60vh;display:flex;flex-direction:column;
  justify-content:center;border-top:1px solid rgba(244,240,232,0.08);
}
.cta-label{font-size:11px;letter-spacing:0.2em;text-transform:uppercase;color:var(--gold);margin-bottom:2rem}
.cta-text{
  font-family:'Bebas Neue',sans-serif;font-size:clamp(48px,8vw,120px);
  line-height:0.9;letter-spacing:0.01em;
}
.cta-link{
  margin-top:4rem;display:inline-flex;align-items:center;gap:1rem;
  font-size:13px;letter-spacing:0.12em;text-transform:uppercase;
  color:var(--gold);text-decoration:none;border-bottom:1px solid var(--gold);
  padding-bottom:4px;width:fit-content;
  transition:gap 0.3s ease;
}
.cta-link:hover{gap:2rem}
const io = new IntersectionObserver(entries=>{
  entries.forEach(e=>{
    if(!e.isIntersecting)return;
    if(e.target.classList.contains('m-inner')){e.target.classList.add('in');return}
    if(e.target.dataset.r){e.target.classList.add('in');return}
  });
},{threshold:0.15,rootMargin:'0px 0px -40px 0px'});

document.querySelectorAll('.m-inner,[data-r]').forEach(el=>io.observe(el));

Search CodeFronts

Loading…