Create a responsive mobile first menu
On This Page:
Part 2 of a series looking at progressively enhanced, mobile first, reponsive navigation
In the first part of this mini series we looked at creating a menu icon for what I believe to be the best solution for your site's mobile navigation.
The outcome was to create an icon by creating an HTML element similar to this -
HTML
Which would then be styled with CSS like this -
CSS
#nav-btn {
background: transparent; border-bottom: 23px double #121212; border-top: 8px solid #121212; height: 7px; width:37px; }
Result
Making an example.
Ok, so let's start making a quick example page with some links that we're going to want to manipulate with this button. I could bore you with the HTML and CSS of this but I'm sure you know how to do that kinda stuff, and if not. You can see it from the resulting JSFiddle below -
Querying your users viewport.
Now for this example, I only want the resulting show/hide menu icon to work on small viewports. Not only do I want to only have this working for small viewports (less than 767px for arguements sake) I was also only want it to work on devices such as an iPhone or Android mobile phone. Therefore whereas we would normally use this kind of media query -
@media only screen and (min-width: 240px) { /* CSS Rules to go here */ }
We are instead going to define a media query as follows that should only affect viewports found on popular mobile phones, like this -
@media only screen and (min-device-width: 1px) and (max-device-width: 767px) { /* CSS Rules to go here */ }
The min-device-width and max-device-width media query rules struggle to work in Android 2.2 so to combat this make sure you've got something similar to this in your HTML
So with this media query we're going to hide the nav element and restyle it.
An updated example (note: this will only look different on your mobile device) -
But first, where's that menu button gone?
In the HTML of the last code example I have not included the DIV for the button. This is because it is going to be added with JavaScript and jQuery. This makes our original HTML cleaner and means your site will be more progressively enhanced and will degrade gracefully when the user has JavaSCript disabled or they're a blackberry user *winks*.
Getting the Javascript going.
So, we're going to add the navigation (menu) button with some JavaScript/jQuery. To do this we're going to use jQuery's prepend, like this -
$('#container').prepend('
');Well that's all well and good, but.
C'mon, c'mon, c'mon, Touch me.
The menu icon is perfect for touch, not so much for a device that'll recognise the media queries but doesn't have touch. So we're going to throw that bit of jQuery into an if statement in JavaScript to only appear if the device has touch. Like this -
if ('ontouchstart' in document.documentElement)
// It's a touch screen device.
// create the button div $('#container').prepend('
'); }else { }
So to tie-in with our device-width media queries this should now show the menu button when viewed on a touch screen device which's viewport is no bigger than 767px.
More Styling? More JavaScript/jQuery?
I want to style the navigation links differently if the device is touch enabled. As much as I have previously said how I dislike a screen full of links having a menu button that shows and hides the navigation is screaming out for nice big links for our chuby fingers to tap, But to progressively enhance this bad boy we only want it changed on devices that have JavaScript enabled. Therefore we're going to add a class the contents #container. Whilst we're at it let's add the show/hide jQuery code too. Like this -
if ('ontouchstart' in document.documentElement) { // It's a touch screen device.
// add claas of touch to container for the CSS $('#container').addClass('touch');
// create the button div $('#container').prepend('
');// make the button div a toggle show/hide$('#nav-btn').click(function(){ $('nav').toggle(); return false; }) }
else { }
Now we add the .touch to the relevant CSS. Here's the updated example.
Having a quick play with this I noticed the dreaded FOUC. So, first I added a 'noJs' class to my HTML then included the following code to swap it with just 'JS' if JavaScript is enabled
After this I changed the display: none; attirbute for the