The off-canvas pattern is a classic approach to responsive navigation. When the viewport is small enough to warrant it, bulky navigation is hidden “off-canvas” and only brought into view when toggled.
Today we’re going to build an off-canvas navigation, using CSS to do the toggling (no JavaScript required) and our good friend Grid to form the page structure. Here’s the full page demo of what we’re working towards.
Basic Page Structure
Let’s begin by building a basic page; we’re aiming for something like this:
Clik here to view.

This is a fairly typical semantic page structure; you’ll see that everything remains in a single column for small viewports, then the aside moves across on larger screens. The <nav>
element is highlighted in blue for clarity.
Here’s our markup:
<header role="banner"><h1>Header</h1></header><nav id="nav" role="navigation"><ul><li><a href="#">Item 1</a></li><li><a href="#">Item 2</a></li><li><a href="#">Item 3</a></li></ul></nav><section role="main"><article><h2>Article</h2><p>Curabitur orci lacus, auctor ut facilisis nec, ultricies quis nibh. Phasellus id diam sollicitudin, malesuada turpis id, gravida erat. Maecenas placerat elit vel hendrerit convallis. Sed in mauris ut justo vulputate viverra feugiat ac dui. Fusce feugiat arcu in vehicula vehicula. Donec varius justo at nulla aliquet volutpat.</p><p>Ut id rutrum eros. Nulla tristique, magna et mattis vulputate, mi eros suscipit turpis, nec bibendum turpis nunc feugiat sapien. Nunc arcu est, lacinia id diam quis, sagittis euismod neque. Nullam fringilla velit sed porta gravida. Proin eu vulputate libero. Ut a lacinia enim. Etiam venenatis mauris et orci tempor congue. Sed tempor eros et ultricies congue. Aenean sed efficitur orci. Nulla vel tempus mi.</p><p>Ut cursus suscipit augue, id sagittis nibh faucibus eget. Etiam suscipit ipsum eu augue ultricies, at rhoncus mi faucibus. In et tellus vitae leo scelerisque fringilla nec at nunc.</p></article></section><aside><h3>Aside</h3></aside><footer><h3>Footer</h3></footer>
Now let’s add some visual styles and some Grid rules.
Gettin’ Griddy With it
Begin by wrapping all our structural elements in a containing element of some kind–this will be our Grid container. I’m using <div class="container"></div>
.
Now add some basic grid styles:
.container { display: grid; grid-template-columns: 1fr; grid-gap: 10px; }
Here we’re declaring that the container should be display: grid;
, that it should have a single column of one fractional unit (not strictly necessary at this point, but we’re added it to be thorough), and that the gap between all grid items should be 10px.
Next add a couple of visual styles to make things a bit clearer:
.container > * { color: #353535; font-size: 1.2em; line-height: 1.5; padding: 20px; background: #d0cfc5; } .container nav { background: #136fd2; } nav ul { list-style: none; margin: 0; padding: 0; } nav a { color: #d0cfc5 } nav a:hover { text-decoration: none; }
Make it Responsive
Let’s add a media query, so that when the viewport hits a certain size (let’s go for 600px) the layout changes.
@media only screen and (min-width: 600px) { /* grid */ .container { grid-template-columns: repeat(4, 1fr); } /* specific item styles */ .container header, .container nav, .container footer { grid-column: span 4; } .container section { grid-column: span 3; } }
So now, on larger screens, the grid declaration changes to grid-template-columns: repeat(4, 1fr);
. This gives us four columns of equal width, so we then have to declare how wide we want each of our structural elements. The header
, nav
, and footer
will all span 4 (span across four columns) whilst the section
will span across just three, leaving a gap of one column for the aside
to fill automatically.
Finally, some styles to alter the way the nav looks:
/* nav styles */ nav ul li { display: inline-block; padding: 0 20px 0 0; }
Here’s what we have so far:
Venturing Off-Canvas
This is a perfect example of how CSS positioning can still work on structural elements, even within a declared Grid. We’re going to grab our nav, remove it from the document flow, and position it off-canvas. The other grid items will fall into place just fine.
Clik here to view.
Begin with another media query. We already have our min-width
query, but this time we only want to style elements up to a max-width
. Up until our viewport reaches that magic 600px we want the nav to be positioned off-canvas:
@media only screen and (max-width: 599px) { #nav { position: fixed; /* or choose `absolute` depending on desired behavior*/ top: 0; bottom: 0; width: 300px; left: -340px; transition: left .3s ease-in-out; } }
We’ve given the nav a fixed width, positioning it left by enough to hide it completely. We used position fixed
too, though you can also use absolute
depending on whether you want the nav to scroll with the window or not.
You’ll also notice the transition
rule, which will take effect once we build some toggle controls.
Toggling
Having made our nav disappear, we now need some controls to bring it back when needed. Let’s add a link to trigger it, and a link to close it again.
Add this to the header:
<a class="toggle open" href="#nav">open</a>
and this to the nav:
<a class="toggle close" href="#">close</a><!-- you might also want to use a “×” -->
We don’t need the open link to be visible on larger screens, so we’ll hide the .toggle elements within our min-width media query:
.toggle { display: none; }
:target
What’s important in the above links is the presence of a “fragment identifier” (the #nav
in the href). These identifiers are used by browsers to navigate directly to specific elements on a page. In this case, we’re targeting whichever element matches the id “nav”, and once it’s been targeted we can style it using the :target
pseudo class. No JavaScript necessary!
Add the following to our max-width media query:
#nav:target { left: 0; }
That’s it! That’s our toggle.
Here’s what we have now:
You’ll need to take a look at the full page demo to full appreciate what it’s doing.
Conclusion
And we’re done! I’ve kept styling down to a minimum so as not to get in the way, but feel free to go nuts and make this look exactly how you want.
You may also prefer a JavaScript toggle, instead of the :target
, in which case you have all the pieces in place to make that work too.
I hope you enjoyed this little Grid exercise, stay tuned for more to come!
Useful Resources
- Understanding CSS Grid Layout series on Tuts+
- Examining Responsive Navigation: Off Canvas Patterns by Steven Bradley
- Subscribe to csslayout.news
Clik here to view.

Clik here to view.
Clik here to view.