By Harry Roberts
Harry Roberts is an independent consultant web performance engineer. He helps companies of all shapes and sizes find and fix site speed issues.
Written by Harry Roberts on CSS Wizardry.
If you follow me on Twitter you’ll know I’ve been pretty enthused about this fluid CSS carousel of mine. There are two aspects to it; the fluidity and the CSS functionality.
The demo features photos of me, taken on various mountains by Suze. Cheers to her for the content.
Making a carousel fluid is actually ridiculously simple. Let us assume you have five panels. Remember that number!
A carousel is made up of three basic components:
Thus, our markup is:
<div class=carousel>
<ul class=panes>
<li>
<h2>Pane 01 title</h2>
<img src=pane-01.jpg alt="">
</li>
<li>
<h2>Pane 02 title</h2>
<img src=pane-02.jpg alt="">
</li>
<li>
<h2>Pane 03 title</h2>
<img src=pane-03.jpg alt="">
</li>
<li>
<h2>Pane 04 title</h2>
<img src=pane-04.jpg alt="">
</li>
<li>
<h2>Pane 05 title</h2>
<img src=pane-05.jpg alt="">
</li>
</ul>
</div>
Now, the viewport defines what we see, we slide our panes behind it and they poke through. We just slide these behind the viewport to show a certain amount at a time, et voilà; carousel. But, you all know how carousels work, right…?
To make this fluid is so simple. The viewport needs to fill its container, so this gets width:100%;
. Easy.
One pane needs to fit nicely in the viewport, so this needs to occupy 100% of the viewport. With this in mind…
We have five panes, remember, so the ul
needs a width of 500%. Five panes that are each the same width as the viewport gives us a width of 500%.
Now we know:
.carousel{
width:100%;
}
.panes{
width:500%;
}
So if .panes
holds five panes, each pane should be 20% its width. This is where it might get a little confusing…
The viewport is 100% width, the wrapper is five times as big as that and each pane is one fifth the width of the wrapper.
Our code is left at:
/*------------------------------------*\
$CAROUSEL
\*------------------------------------*/
.carousel{
overflow:hidden;
width:100%;
}
.panes{
list-style:none;
position:relative;
width:500%; /* Number of panes * 100% */
overflow:hidden; /* This is used solely to clear floats, it does not add functionality. */
-moz-animation:carousel 30s infinite;
-webkit-animation:carousel 30s infinite;
animation:carousel 30s infinite;
}
.panes > li{
position:relative;
float:left;
width:20%; /* 100 / number of panes */
}
.carousel img{
display:block;
width:100%;
max-width:100%;
}
.carousel h2{
font-size:1em;
padding:0.5em;
position:absolute;
right:10px;
bottom:10px;
left:10px;
text-align:right;
color:#fff;
background-color:rgba(0,0,0,0.75);
}
The basic equation for making a carousel with any number of panes is:
<pre><code>.carousel{
width:100%;
}
.panes{
width:<mark>100 * number of panes</mark>%;
}
.panes > li{
width:<mark>100 / number of panes</mark>%;
}</code></pre>
So, a four-pane carousel would be:
<pre><code>.carousel{
width:100%;
}
.panes{
width:<mark>400</mark>%;
}
.panes > li{
width:<mark>25</mark>%;
}</code></pre>
It really is that simple. That’s all there is to making a fluid carousel.
Okay, in this carousel I decided I was going to power it with CSS. This is super unorthodox so if you’re yelling WTF at your screen please read on!
It was going to be (and actually is) used on my good friend Sam Penrose’s new design portfolio, so knowing we had free reign and a little chance to experiment I decided to opt for a pure CSS solution.
This is simple in theory but the maths gets so tricky.
All we do is animate .panels
from right to left then back again. We animate for a bit, we pause, we animate again, pause again and so on until it’s done. Then we loop it infinitely.
The CSS is:
@keyframes carousel{
0% { left:0; }
11% { left:0; }
12.5% { left:-100%; }
23.5% { left:-100%; }
25% { left:-200%; }
36% { left:-200%; }
37.5% { left:-300%; }
48.5% { left:-300%; }
50% { left:-400%; }
61% { left:-400%; }
62.5% { left:-300%; }
73.5% { left:-300%; }
75% { left:-200%; }
86% { left:-200%; }
87.5% { left:-100%; }
98.5% { left:-100%; }
100% { left:0; }
}
The numbers are so tricky to work with and if anyone can come up with a decent equation to describe it I would be so happy. I’ve got Nick on the job, but I’ve not worked it out yet; I can’t quite grasp the relation between five panes and the numbers above in a way that I could do some quick maths to work out the animations for a four-pane carousel.
The problem is that you have to know how many full moves and pauses are needed for a full iteration of the carousel (before it starts on its infinite loop), and then how to evenly space these numbers between 0 and 100%. My animations last for 1.5% and pause for 11%, these numbers are perfect for adding up to 100%.
Massive thanks to Clay who’s worked out that the number of steps is 4n-3 and that the total time between start of one animation and the next is 100 / 2(n-1) (where n is the number of panes). See his full comment.
But, the code above will sort you out a perfect animation for a five panes so feel free to nab it! I’ll update if and when I crack some maths!
Using CSS for this is really, really unorthodox so comes with some massive caveats.
Do not use CSS to animate this carousel if:
If you are going to use this CSS-only method then ensure that:
If you can’t use CSS then combine the fluidity above with plain ol’ trusty JS.
I will happily say that the fluidity is the most important, useful and impressive thing about this technique. Until I, or anyone, can get you a decent equation to substitute your numbers into, the CSS animations are too cumbersome and restrictive to be of large-scale use to most people.
Harry Roberts is an independent consultant web performance engineer. He helps companies of all shapes and sizes find and fix site speed issues.
Hi there, I’m Harry Roberts. I am an award-winning Consultant Web Performance Engineer, designer, developer, writer, and speaker from the UK. I write, Tweet, speak, and share code about measuring and improving site-speed. You should hire me.
You can now find me on Mastodon.
I am available for hire to consult, advise, and develop with passionate product teams across the globe.
I specialise in large, product-based projects where performance, scalability, and maintainability are paramount.