marknormanfrancis.com

Why optimising CSS is dangerous

Saturday 27 March, 2010 at 10.20pm

updated
Saturday 12 June, 2021 at 8.54am
(Fixing broken links)

One of the basic principles that an effective web developer has to learn is the cascade part of cascading style sheets. “So … explain the cascade to me.” is still my favourite question to ask when interviewing candidates for web development jobs (and if you don’t sweat, stammer and fumble your explanation, the chances are you really don’t understand it).

If the idea of letting an automated tool rewrite your CSS and potentially change the source order doesn’t doesn’t fill you with The Fear™, then you definitely don’t understand the cascade properly.

The last rule of determining how CSS is applied to elements is that if two declarations have equal weight, origin and specificity, then the last rule wins.

This is why the following CSS would colour an <h1> element blue:

h1 { color: red; }
h1 { color: blue; }

They are otherwise equal statements, so the last one specified is used to style the element.

Alphabetical order

Imagine you are using the common technique of using sprited images to provide icons to a list of links to bookmarking sites. Your CSS might look like:

#bookmarks .item { 
    background: url(sprite.png)
                no-repeat
                0 0;
}
#bookmarks .delicious {
    background-position: 0 -15px;
}
#bookmarks .pinboard {
    background-position: 0 -45px;
}
#bookmarks .magnolia {
    background-position: 0 -30px;
}

Running that code through a CSS optimiser that output the modified rules in alphabetical order would give you:

#bookmarks .delicious{background-position:0 -15px;}
#bookmarks .item{background:url(sprite.png) no-repeat 0 0;}
#bookmarks .magnolia{background-position:0 -30px;}
#bookmarks .pinboard{background-position:0 -45px;}

This would reset the background-position property for the item with the class delicious, giving it the wrong part of the sprite image. Of course, this is a slightly contrived example. But it illustrates the dangers of arbitrarily re-ordering code where order is significant.

The real solution is to use #bookmarks li instead of #bookmarks .item because then the class has greater specificity, regardless of the order it appears in the output style sheet.

An explicit output order

After optimisation, my CSS Prepare code will group the rule sets in the order elements, then IDs, then classes. And inside of each group, the rule sets are indeed output in alphabetical order. Whilst I could output them randomly, a guaranteed ordering makes testing the code much easier.

If you are writing your CSS well and using specificity meaningfully, this reordering should result in zero actual changes to the rendered document if you were to use CSS Prepare. But you should always test the output of any minifier before putting code live. Doubly so something that has the potential to change the meaning of your code.