30 CSS Hover Effects 07 / 30
CSS Magnetic Liquid Button Hover Effect
Four button hover effects with fluid, organic transitions — liquid fill bubble, morphing border-radius blob, surface tension ripple, and stretched rubber — using border-radius animation and pseudo-element scale transforms for a tactile, material-feeling interaction.
This is a full-page demo — interact inside the frame above, or open it in the playground for the full-screen experience.
The code
<div class="hv-07">
<div class="hv-07__grid">
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--fill">
<span>Liquid Fill</span>
</button>
<span class="hv-07__label">scale-Y fill</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--blob">
<span>Blob Morph</span>
</button>
<span class="hv-07__label">border-radius keyframe</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--tension">
<span>Surface Tension</span>
</button>
<span class="hv-07__label">ellipse clip-path burst</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--rubber">
<span>Rubber Stretch</span>
</button>
<span class="hv-07__label">squash-and-stretch</span>
</div>
</div>
</div> <div class="hv-07">
<div class="hv-07__grid">
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--fill">
<span>Liquid Fill</span>
</button>
<span class="hv-07__label">scale-Y fill</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--blob">
<span>Blob Morph</span>
</button>
<span class="hv-07__label">border-radius keyframe</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--tension">
<span>Surface Tension</span>
</button>
<span class="hv-07__label">ellipse clip-path burst</span>
</div>
<div class="hv-07__cell">
<button class="hv-07__btn hv-07__btn--rubber">
<span>Rubber Stretch</span>
</button>
<span class="hv-07__label">squash-and-stretch</span>
</div>
</div>
</div>.hv-07,.hv-07 *,.hv-07 *::before,.hv-07 *::after{box-sizing:border-box;margin:0;padding:0}
.hv-07 ::selection{background:#10b981;color:#fff}
.hv-07{
--bg:#030d09;
--text:#d1fae5;
--dim:#374151;
--green:#10b981;
--teal:#0d9488;
--emerald:#059669;
--mint:#6ee7b7;
font-family:'Segoe UI',system-ui,sans-serif;
background:var(--bg);
min-height:100vh;
display:flex;align-items:center;justify-content:center;
padding:60px 24px;
}
.hv-07__grid{
display:grid;grid-template-columns:repeat(2,1fr);gap:48px;
max-width:720px;width:100%;
}
.hv-07__cell{
display:flex;flex-direction:column;align-items:center;gap:20px;
padding:48px 32px;
background:rgba(255,255,255,.025);
border:1px solid rgba(255,255,255,.07);
border-radius:14px;
}
.hv-07__label{font-size:11px;letter-spacing:.15em;text-transform:uppercase;color:var(--dim)}
/* shared btn base */
.hv-07__btn{
position:relative;overflow:hidden;
padding:14px 36px;
border:2px solid var(--green);
background:transparent;
color:var(--green);
font-size:1rem;font-weight:600;letter-spacing:.06em;
border-radius:8px;cursor:pointer;
transition:color .35s;
}
.hv-07__btn span{position:relative;z-index:1}
.hv-07__btn::before{content:'';position:absolute;inset:0;z-index:0}
/* 1 — liquid fill */
.hv-07__btn--fill::before{
background:var(--green);
transform:scaleY(0);transform-origin:bottom;
transition:transform .35s cubic-bezier(.4,0,.2,1);
}
.hv-07__btn--fill:hover::before{transform:scaleY(1)}
.hv-07__btn--fill:hover{color:#fff}
/* 2 — blob morph */
.hv-07__btn--blob{
border-color:var(--teal);color:var(--teal);
border-radius:8px;
transition:border-radius .5s cubic-bezier(.4,0,.2,1),color .35s,background .35s;
}
.hv-07__btn--blob::before{
background:var(--teal);
border-radius:8px;
transform:scale(0);
transition:transform .35s cubic-bezier(.4,0,.2,1),border-radius .5s;
}
.hv-07__btn--blob:hover{
color:#fff;
border-radius:60% 40% 70% 30% / 40% 70% 30% 60%;
}
.hv-07__btn--blob:hover::before{
transform:scale(1);
border-radius:60% 40% 70% 30% / 40% 70% 30% 60%;
}
/* 3 — surface tension (expanding emerald droplet from button center).
The original used negative percentage margins (margin:-100% 0 0
-100%) to center the ::after, but percentage margins resolve
against the PARENT'S WIDTH on BOTH axes (not against the
element's own size, and NOT against parent's height for
top/bottom margins). On a wide button, margin-top:-100%
overshoots by ~200px upward, pushing the entire droplet above
the visible button area where overflow:hidden clips it to
nothing.
Fix: use transform translate(-50%, -50%) for the centering
(translate-% uses the ELEMENT's own dimensions) and combine the
scale in the same transform. The animated value is the scale
inside the transform, so we declare the centered translate +
scaled-to-zero state, and the :hover handler bumps the scale
portion to 1 while the centering translate stays the same. */
.hv-07__btn--tension{
border-color:var(--emerald);color:var(--emerald);
}
.hv-07__btn--tension::after{
content:'';position:absolute;
top:50%;left:50%;
width:200%;height:200%;
background:var(--emerald);
border-radius:50%;
transform:translate(-50%,-50%) scale(0);
transition:transform .5s cubic-bezier(.4,0,.2,1);
z-index:0;
}
.hv-07__btn--tension:hover::after{transform:translate(-50%,-50%) scale(1)}
.hv-07__btn--tension:hover{color:#fff}
/* 4 — rubber stretch */
.hv-07__btn--rubber{
border-color:var(--mint);color:var(--mint);
transition:transform .2s cubic-bezier(.4,0,.2,1),color .35s,background .35s,border-color .35s;
}
.hv-07__btn--rubber::before{
background:var(--mint);
transform:scaleY(0);transform-origin:bottom;
transition:transform .35s;
}
.hv-07__btn--rubber:hover{
transform:scaleX(1.06) scaleY(0.93);
color:#000;
}
.hv-07__btn--rubber:hover::before{transform:scaleY(1)}
.hv-07__btn--rubber:active{
transform:scaleX(0.97) scaleY(1.04);
}
@media(max-width:520px){.hv-07__grid{grid-template-columns:1fr}}
@media(prefers-reduced-motion:reduce){
.hv-07__btn,.hv-07__btn::before,.hv-07__btn::after{transition:none!important;animation:none!important}
} .hv-07,.hv-07 *,.hv-07 *::before,.hv-07 *::after{box-sizing:border-box;margin:0;padding:0}
.hv-07 ::selection{background:#10b981;color:#fff}
.hv-07{
--bg:#030d09;
--text:#d1fae5;
--dim:#374151;
--green:#10b981;
--teal:#0d9488;
--emerald:#059669;
--mint:#6ee7b7;
font-family:'Segoe UI',system-ui,sans-serif;
background:var(--bg);
min-height:100vh;
display:flex;align-items:center;justify-content:center;
padding:60px 24px;
}
.hv-07__grid{
display:grid;grid-template-columns:repeat(2,1fr);gap:48px;
max-width:720px;width:100%;
}
.hv-07__cell{
display:flex;flex-direction:column;align-items:center;gap:20px;
padding:48px 32px;
background:rgba(255,255,255,.025);
border:1px solid rgba(255,255,255,.07);
border-radius:14px;
}
.hv-07__label{font-size:11px;letter-spacing:.15em;text-transform:uppercase;color:var(--dim)}
/* shared btn base */
.hv-07__btn{
position:relative;overflow:hidden;
padding:14px 36px;
border:2px solid var(--green);
background:transparent;
color:var(--green);
font-size:1rem;font-weight:600;letter-spacing:.06em;
border-radius:8px;cursor:pointer;
transition:color .35s;
}
.hv-07__btn span{position:relative;z-index:1}
.hv-07__btn::before{content:'';position:absolute;inset:0;z-index:0}
/* 1 — liquid fill */
.hv-07__btn--fill::before{
background:var(--green);
transform:scaleY(0);transform-origin:bottom;
transition:transform .35s cubic-bezier(.4,0,.2,1);
}
.hv-07__btn--fill:hover::before{transform:scaleY(1)}
.hv-07__btn--fill:hover{color:#fff}
/* 2 — blob morph */
.hv-07__btn--blob{
border-color:var(--teal);color:var(--teal);
border-radius:8px;
transition:border-radius .5s cubic-bezier(.4,0,.2,1),color .35s,background .35s;
}
.hv-07__btn--blob::before{
background:var(--teal);
border-radius:8px;
transform:scale(0);
transition:transform .35s cubic-bezier(.4,0,.2,1),border-radius .5s;
}
.hv-07__btn--blob:hover{
color:#fff;
border-radius:60% 40% 70% 30% / 40% 70% 30% 60%;
}
.hv-07__btn--blob:hover::before{
transform:scale(1);
border-radius:60% 40% 70% 30% / 40% 70% 30% 60%;
}
/* 3 — surface tension (expanding emerald droplet from button center).
The original used negative percentage margins (margin:-100% 0 0
-100%) to center the ::after, but percentage margins resolve
against the PARENT'S WIDTH on BOTH axes (not against the
element's own size, and NOT against parent's height for
top/bottom margins). On a wide button, margin-top:-100%
overshoots by ~200px upward, pushing the entire droplet above
the visible button area where overflow:hidden clips it to
nothing.
Fix: use transform translate(-50%, -50%) for the centering
(translate-% uses the ELEMENT's own dimensions) and combine the
scale in the same transform. The animated value is the scale
inside the transform, so we declare the centered translate +
scaled-to-zero state, and the :hover handler bumps the scale
portion to 1 while the centering translate stays the same. */
.hv-07__btn--tension{
border-color:var(--emerald);color:var(--emerald);
}
.hv-07__btn--tension::after{
content:'';position:absolute;
top:50%;left:50%;
width:200%;height:200%;
background:var(--emerald);
border-radius:50%;
transform:translate(-50%,-50%) scale(0);
transition:transform .5s cubic-bezier(.4,0,.2,1);
z-index:0;
}
.hv-07__btn--tension:hover::after{transform:translate(-50%,-50%) scale(1)}
.hv-07__btn--tension:hover{color:#fff}
/* 4 — rubber stretch */
.hv-07__btn--rubber{
border-color:var(--mint);color:var(--mint);
transition:transform .2s cubic-bezier(.4,0,.2,1),color .35s,background .35s,border-color .35s;
}
.hv-07__btn--rubber::before{
background:var(--mint);
transform:scaleY(0);transform-origin:bottom;
transition:transform .35s;
}
.hv-07__btn--rubber:hover{
transform:scaleX(1.06) scaleY(0.93);
color:#000;
}
.hv-07__btn--rubber:hover::before{transform:scaleY(1)}
.hv-07__btn--rubber:active{
transform:scaleX(0.97) scaleY(1.04);
}
@media(max-width:520px){.hv-07__grid{grid-template-columns:1fr}}
@media(prefers-reduced-motion:reduce){
.hv-07__btn,.hv-07__btn::before,.hv-07__btn::after{transition:none!important;animation:none!important}
}How this works
The liquid fill effect places a ::before pseudo-element inside the button, scaled to scaleY(0) by default with transform-origin: bottom. On hover it scales to scaleY(1) with cubic-bezier(.4,0,.2,1) easing. The blob morph animates border-radius through multiple percentage pairs — from a rounded rectangle to an asymmetric organic shape — using an @keyframes that runs only on :hover.
The rubber-stretch variant combines scaleX(1.06) scaleY(0.94) on mouseenter timing and then scaleX(0.97) scaleY(1.03) as a bounce-back, creating a squash-and-stretch feel using pure CSS transition with back-easing. Surface tension adds a clip-path: ellipse() on ::after that expands from the center like a droplet impact.
Customize
- Adjust fill speed by changing the
::beforetransition duration —.25sfeels snappy;.5sfeels more viscous and slow. - Change fill color by editing
background: var(--fill)on the::before— try a semi-transparent color for a tinted-glass liquid effect. - Add a subtle shadow on hover with
box-shadow: 0 8px 30px rgba(0,0,0,.3)to lift the button off the page as it fills. - Modify the blob morph shape by editing the target
border-radiuskeyframe values — extreme values like70% 30% 60% 40% / 40% 60% 70% 30%produce amoeba shapes. - Combine the fill with a color transition on
colorso the text switches from dark to light as the fill covers the button.
Watch out for
- The fill
::beforemust sit behind the button label — ensurez-index: 0on::beforeandposition: relative; z-index: 1on the text span. - Safari clips pseudo-element transforms at the parent's
overflow: hiddenboundary — this is desired here but can conflict if you add a visiblebox-shadowon the button itself. border-radiuskeyframe animation causes the browser to repaint (not just composite) — limit concurrent blob animations to avoid paint jank on low-powered devices.
Browser support
| Chrome | Safari | Firefox | Edge |
|---|---|---|---|
| 60+ | 13+ | 60+ | 60+ |
All effects use transitions and keyframes that are universally supported — no experimental flags needed.