Monday 10 March 2008

CSS Expression Optimization

 

CSS Expressions

CSS expressions were introduced in Internet Explorer 5.0 and it allows you to to assign a JavaScript expression to a CSS property. For example, this allows you to set the position of an element according to the browser size.

#myDiv {
position: absolute;
width: 100px;
height: 100px;
left: expression(document.body.offsetWidth - 110 + "px");
top: expression(document.body.offsetHeight - 110 + "px");
background: red;
}


Not the best way to achieve this but it is easy enough to understand.



Dynamic Expressions



Not only does CSS expressions allow you to calculate the CSS value inside the style declarations, it also recalculates the expression so that property value is always correct. This means that the expression needs to be recalculated every time the document.body.offsetWidth is changed. If this was the case dynamic expressions would actually be a lot more useful. This isn't the case and recalculations of the expressions happens every time a JS execution thread is finished. This means that if you move your mouse and have a listener to onmousemove all the dynamic expressions are recalculated. This does not require you to be a genious to figure out that this can bring the simplest web application to a crawl.



Take a look at the following CSS block:



#myDiv {
border: 10px solid Red;
width: expression(ieBox ? "100px" : "80px");
}


Asuming ieBox is a constant flag which is true when IE is in quirks mode, the expression will at all times result in "80px" (or "100px"). Even though the expression is constant it will be recalculated a lot of times. The question is how to remove these extra calculations for constant expressions.



Constant Expressions



The thing to do is to go through all the style sheet declarations and replace the constant expressions with a constant value. In the previous example, assuming we are using IE6 in standard mode, we want:



#myDiv {
border: 10px solid Red;
width: 80px;
}


So, how do we find out if an expression is constant. The easiest way is to mark the expression so that it can be easily found. The solution we are going to use is to enclose the expression in a function that we know the name of and is equal to the identity function.



function constExpression(x) {
return x;
}


And then in the CSS block we use:



#myDiv {
border: 10px solid Red;
width: expression(constExpression(ieBox ? "100px" : "80px"));
}


Usage



The first thing you need to do is to include the cssexpr.js file. This should be done before any usage of constExpression.



<script type="text/javascript" src="cssexpr.js"></script>


After this you can use constExpression in any style block or any file included using a link element or using the @import directive in a these. Notice that style attributes are not checked due to performance reasons.



Implementation



The idea here is to go through all style sheets and then go through all rules and finally go through all the cssText. To do this we first go through the document.styleSheets collection.



function simplifyCSSExpression() {
try {
var ss,sl, rs, rl;
ss = document.styleSheets;
sl = ss.length

for (var i = 0; i < sl; i++) {
simplifyCSSBlock(ss[i]);
}
}
catch (exc) {
alert("Got an error while processing css. The page " +
"should still work but might be a bit slower");
throw exc;
}
}


In the style sheet we go thorugh the imports collection and then the rules collection. We also check that the cssText contains "expression(constExpression(" so that we do not process the style sheet in vain.



function simplifyCSSBlock(ss) {
var rs, rl;

// Go through imports
for (var i = 0; i < ss.imports.length; i++)
simplifyCSSBlock(ss.imports[i]);

// if no constExpression we don't have to continue
if (ss.cssText.indexOf("expression(constExpression(") == -1)
return;

rs = ss.rules;
rl = rs.length;
for (var j = 0; j < rl; j++)
simplifyCSSRule(rs[j]);
}


Now we go through the cssText of each rule and update the text, using a function named simplifyCSSRuleHelper, until the text does not change any more.



function simplifyCSSRule(r) {
var str = r.style.cssText;
var str2 = str;
var lastStr;

// update string until the updates does not change the string
do {
lastStr = str2;
str2 = simplifyCSSRuleHelper(lastStr);
} while (str2 != lastStr)

if (str2 != str)
r.style.cssText = str2;
}


The helper function just finds the first expression and evaluates that and replaces the expression with the value.



function simplifyCSSRuleHelper(str) {
var i, i2;
i = str.indexOf("expression(constExpression(");
if (i == -1) return str;
i2 = str.indexOf("))", i);
var hd = str.substring(0, i);
var tl = str.substring(i2 + 2);
var exp = str.substring(i + 27, i2);
var val = eval(exp)
return hd + val + tl;
}


Finally we need to call the function simplifyCSSExpression and this is done once the page is loaded.



if (/msie/i.test(navigator.userAgent) && window.attachEvent != null) {
window.attachEvent("onload", function () {
simplifyCSSExpression();
});
}


 


Download



Author: Erik Arvidsson

Share:

0 comments:

Post a Comment