This is a simple solution to a responsive problem I was having recently - in essence it's similar to a problem Jeremy Keith wrote about back in April. I needed my JavaScript to recognise which media queries were currently being fired, so that it could in turn execute some related functionality. This also needed to run when the browser was resized, firing when different min-width conditions were met. So in context, this could be binding a click event to open a lightbox at larger breakpoints, and then unbinding this event when we hit lower widths.
This should be pretty simple; surely just compare the media query width to the current window width in JavaScript? Unfortunately, differing ideas of what true window size should be - Firefox includes the vertical scrollbar in the total window size, whereas webkit browsers don't - means checking for a window width of 800px will not always give consistent results across browsers.
For anyone lucky enough to be working on projects that don't suport IE8<, MatchMedia takes care of this problem (with a fix for resize support). Unfortunately, I needed to support down to IE7 and there isn't a polyfill to patch MatchMedia with resizing for IE7/8 as yet.
Having previously read Jeremy's blog post, I knew that a version of what I wanted existed using pseudo elements. It wasn't until speaking to a friend of mine, Rich](https://twitter.com/middric), about the problem, that he suggested manipulating the z-index of an arbitrary div to perform this role. This is the resulting solution that came out of his suggestion.
First, create the div element. As Jeremy's points out in his blog post, a z-index applied to the body wouldn't work - webkit always returns a value of auto in this case - but doing so on an element within the body allows the JavaScript to retrieve the z-index value as we need to. This is added with JS, to keep the solution out of the markup:
var breakpointTest = document.createElement('div');
breakpointTest.setAttribute('id', 'breakpoint-test');
document.body.appendChild(breakpointTest);
Next, we add a set of media queries that manipulate the z-index of our div element. Rather than using arbitrary numbers to represent the breakpoints, setting the z-index to the min-width of the media query makes our JavaScript checks make more sense when read in isolation.
So as an example here, we'll set checks for 480px and 850px:
#breakpoint-test {
z-index:1;
position:relative;
}
@media all and (min-width: 480px) {
#breakpoint-test {
z-index:480;
}
}
@media all and (min-width: 850px) {
#breakpoint-test {
z-index:850;
}
}
...
So, we've now got a div that will have a z-index equal to the min-width value of the currently active media query. All that's left is to check for this value back in the JavaScript. I'll do this on resize, but you could just do this on page load if you don't need it to fire on window resize. Note that although I'm not throttling my resize event here to keep the example concise, it's a good idea to do this.
var lastminwidth = 0, //keep track of last fired minwidth
minwidth = 1; //the current minwidth
window.onresize = function(event){
minwidth = window.getComputedStyle(document.getElementById('breakpoint-test'), null).getPropertyValue("z-index");
if (lastminwidth !== minwidth) {
if (minwidth >= 480) {
//fire js bindings for over 480px
} else if ( minwidth < 480 ) {
//fire js bindings for under 480px
}
}
}
window.fireEvent('resize'); //trigger resize on load for first run
We keep a track of our lastminwidth to ensure checks only happen when our media queries actually change, rather than every time the window resizes. One caveat here is that getComputedStyle isn't supported by lower versions of IE. You have two optons to solve this; you can patch it, or if you're using jQuery, you can return the z-index using $('#breakpoint-test').css('zIndex')
instead.
This now lets you fire different functions based on your minwidth checks. I think it's a fairly elegant cross-browser solution, but if you have any ideas as to how it can be improved, let me know in the comments.
Article posted on the 24th September 2012