DOM Update Followed By A Big Loop Doesn't Render In Time
Solution 1:
function do_rebuild() {
for (var i=0;i<9000;i++) {
// append element...
}
hideOverlay();
}
function rebuild() {
showOverlay(); // The overlay will appear
window.setTimeout('do_rebuild();',1);
}
Is the only cross-browser way I know of.
Solution 2:
You need to put your loop inside a set timeout so that it doesn't hold up the page. Even if your overlay is displayed, nobody likes their page freezing
var counter = 0;
function rebuild() {
showOverlay();
doWork();
}
function doWork() {
if(counter < 9000){
// append element
counter++;
setTimeout(function(){
doWork();
},10);
}
else {
hideOverlay();
}
}
EDIT: This answer will actually take significantly longer to process the page though. Somewhere in the realm of 90 seconds which is pretty unacceptable, the other alternative could be to set a timeout every 100 iterations, which will add about 1 sec to the total load time, but should stop the page from freezing.
function doWork() {
if(counter < 9000){
// append element
if(counter % 100 == 0) {
setTimeout(function(){
doWork();
},10);
counter++;
}
else {
doWork();
counter++;
}
}
else {
hideOverlay();
}
}
Solution 3:
The other answers show how to do this in a "pseudo-threading" way, using timers.
In the meantime I have learned about Web Workers, which are a threading solution that separates script execution from DOM updates.
They are currently supported in WebKit and Firefox, and support is planned for IE 10.
Solution 4:
you could try this:
var loopCount = 0,
build = function() {
var frgm = document.createDocumentFragment();
for (var i = 0; i < 200 && loopCount < 9000; i++) {
// some codes
frgm.appendChild(someElement);
loopCount ++;
}
parentNode.appendChild(frgm);
if (loopCount < 9000) {
setTimeout(build, 10);
}
};
function rebuild() {
showOverlay(); // The overlay never appears...
build();
hideOverlay();
}
Post a Comment for "DOM Update Followed By A Big Loop Doesn't Render In Time"