CSS Only Dropdown Menu
Earlier in my dev career I leaned heavily on jQuery to get me what I needed. In a situation like this, I would probably have used the
slideDown() function or something similar.
Now we use plain old vanilla JS to get us what we need. Let me show you how to make a pretty nice looking drop-down navigation using mostly CSS. The secret, max-height transition.
See the Pen Simple Nav Dropdown with css transition by Ted Krueger (@PhiloBeddoe) on CodePen.
Notice in the example above, when you click Dropdown Action, the dropdown menu displays. The dropdown has a
max-height: 0; by default. After the click, we add a class of
active to the dropdown and change the
max-height property. I set it to be
100vh when active. Remember, this is just a
max-height so by doing this I’m saying the dropdown is going to be at most the height of the window. From here you could add some overflow styles to ensure that users could scroll so they see all the content within the dropdown, or you could change the number. The important part is that the number does need to be specific. Hopefully, this is something we see in the future for CSS, but for now
max-height it is.
To ensure the transition is nice and smooth as the menu opens and closes, we’ll add
ease-in-out for our
transition-timing-function. The code will look like this:
transition: max-height var(--speed) ease-in-out;
var(--speed) is set above as
0.3s. The timing function is key to the menu animating nice as it opens and closes. If we had set
ease-in then it would open nicely, but “snap” shut when you close the menu.
When thinking about how I would do this, the first thing I thought of was using a checkbox. That seems like the only way that you could still target a state in your code and trigger the max-height transition of your menu.
I’m not sure about the accessibility of having your mobile menu fire with a checkbox rather than a button, but that’s not really the point I’m trying to make. What I’m trying to say is, think outside the box and push the limits of what you can do with CSS. You can use height CSS transitions to get the same look that you could get by using
.slideDown() (shout out to all my jQuery devs).
Check out the working example below.
See the Pen css only drop down nav by Ted Krueger (@PhiloBeddoe) on CodePen.
The CSS here might seem a little more complicated regarding the
&:checked + label + .dropdown but this is necessary. Also, note the structure of the HTML. You’ll want to be sure that the
site-nav are siblings. This makes it easier to target the site-nav when the dropdown-action is checked. Plus, you wouldn’t want your menu to be wrapped by a label. We can use this structure in the dropdown menu for our submenus as well.
<li class="site-nav__item"> <input type="checkbox" id="dropdown-menu" value="Dropdown Menu" class="dropdown-toggle"> <label for="dropdown-menu" class="site-nav__link dropdown-action" aria-label="Toggle Dropdown"> <span class="dropdown-toggle-text">Learn</span> </label> <ul class="dropdown"> <li class="dropdown__item"> <a href="#" class="site-nav__link">People</a> </li> <li class="dropdown__item"> <a href="#" class="site-nav__link">Places</a> </li> <li class="dropdown__item"> <a href="#" class="site-nav__link">Things</a> </li> </ul> </li>
I think this stuff is pretty cool, and I hope you do too. Even if you don’t, that’s fine. I’m just happy you made it this far in the post. Anyways, I think it’s important to push the limits of what we think is capable of CSS, or any language for that matter. Get out of your comfort zone and try some new stuff.