I have a web page that is injecting HTML delivered via AJAX. I don't know, a priori, the size of the content, but I need to position the content on the page based on its size. Here's a simple approach that doesn't work:
- Set parent container's opacity to 0.
- Insert the content into the DOM.
- Measure the sizes, e.g.
$el.outerWidth(true)
. - Position the content based on those sizes.
- Set the parent container's opacity (back) to 1.0.
The problem with this approach is that the content is quite plex and includes images. When outerWidth()
is called, the browser hasn't yet finished repainting the screen so that the sizes returned aren't accurate. (Interestingly, in my case the initial dimensions are always significantly larger than the ultimate values, not 0 as I might expect.)
I can reduce the problem by setting a timer between steps 2 and 3, but no matter how long I make the timer, there will surely be some browser on some system that will exceed it. And if I make the timer long enough to cover most cases, that will penalize users with faster systems by introducing unnecessary delays. Although a few of the answers referenced below suggest a setTimeout(fn, 0)
is sufficient, I've found that not to be be the case. I've measured paint times as high as 60 ms on a 2012 MacBook Air, and one answer below suggests 500 ms.
It looks like Mozilla at one time had an onPaint
event that might have solved my problem … if it still existed and if other browsers had adopted it. (Neither of which is the case.) And Mutation Observers don't seem to account for paint time when reporting changes in elements.
Solutions or ments very much appreciated. As noted below, this has been asked before but most of the questions are at least a year old and I'm desperate enough to try again.
Related questions that don't, unfortunately, offer a workable answer
- How to detect when an image has finished rendering in the browser (i.e. painted)?
- jQuery - How to execute script as soon as we know true height (with images) of ajax-loaded element?
- DIV width immediately after append is calculating wrong?
- jQuery returns height of 0 for div element immediately after appending element
- Javascript: unable to get height of newly created element
- jQuery: Why does .width() sometimes return 0 after inserting elements with .html()?
Update: Well, it appears there simply isn't a general solution for this problem. In my specific case, I found a property that would reliably change once the content was loaded and layout was plete. The JavaScript polls for a change in this property. Not at all elegant, but it was the best option I could find.
I have a web page that is injecting HTML delivered via AJAX. I don't know, a priori, the size of the content, but I need to position the content on the page based on its size. Here's a simple approach that doesn't work:
- Set parent container's opacity to 0.
- Insert the content into the DOM.
- Measure the sizes, e.g.
$el.outerWidth(true)
. - Position the content based on those sizes.
- Set the parent container's opacity (back) to 1.0.
The problem with this approach is that the content is quite plex and includes images. When outerWidth()
is called, the browser hasn't yet finished repainting the screen so that the sizes returned aren't accurate. (Interestingly, in my case the initial dimensions are always significantly larger than the ultimate values, not 0 as I might expect.)
I can reduce the problem by setting a timer between steps 2 and 3, but no matter how long I make the timer, there will surely be some browser on some system that will exceed it. And if I make the timer long enough to cover most cases, that will penalize users with faster systems by introducing unnecessary delays. Although a few of the answers referenced below suggest a setTimeout(fn, 0)
is sufficient, I've found that not to be be the case. I've measured paint times as high as 60 ms on a 2012 MacBook Air, and one answer below suggests 500 ms.
It looks like Mozilla at one time had an onPaint
event that might have solved my problem … if it still existed and if other browsers had adopted it. (Neither of which is the case.) And Mutation Observers don't seem to account for paint time when reporting changes in elements.
Solutions or ments very much appreciated. As noted below, this has been asked before but most of the questions are at least a year old and I'm desperate enough to try again.
Related questions that don't, unfortunately, offer a workable answer
- How to detect when an image has finished rendering in the browser (i.e. painted)?
- jQuery - How to execute script as soon as we know true height (with images) of ajax-loaded element?
- DIV width immediately after append is calculating wrong?
- jQuery returns height of 0 for div element immediately after appending element
- Javascript: unable to get height of newly created element
- jQuery: Why does .width() sometimes return 0 after inserting elements with .html()?
Update: Well, it appears there simply isn't a general solution for this problem. In my specific case, I found a property that would reliably change once the content was loaded and layout was plete. The JavaScript polls for a change in this property. Not at all elegant, but it was the best option I could find.
Share Improve this question edited May 23, 2017 at 12:33 CommunityBot 11 silver badge asked Nov 2, 2013 at 0:54 Stephen ThomasStephen Thomas 14.1k2 gold badges34 silver badges53 bronze badges 3-
What about adding it to the DOM with
display:none
, deferring execution (with asetTimeout
) and then arranging things based on its height/width ? – Benjamin Gruenbaum Commented Nov 2, 2013 at 1:00 -
Isn't the
load
event supposed to fire up once an image has been downloaded? And aren't the dimensions of the image known once retrieved? Maybe render the image outside of the viewport before including it where you need it to make sure the browser actually calculates dimensions (also considering CSS rules). – Kiruse Commented Nov 2, 2013 at 1:07 - Wait until gap between last ResizeObserver event? – Nikos Commented Jul 27, 2023 at 7:49
2 Answers
Reset to default 4I've started looking at the current situation of this and my solution would be this:
doSomeDomManipulation();
requestAnimationFrame(() => {
setTimeout(() => {
alert("finished painting");
}, 0);
});
Source for this approach:
This used to be inside a note on the old MDN page on the topic of MozAfterPaint:
"Web pages that want to take an action after a repaint of the page can use requestAnimationFrame with a callback that sets a timeout of zero to then call the code that takes the desired post-repaint action."
Edit: An even better version that uses queueMicrotask
instead of setTimeout
(setTimeout
queues a macro task, which would run later):
doSomeDomManipulation();
requestAnimationFrame(() => {
queueMicrotask(() => {
alert("finished painting");
});
});
You could see if window load will do the trick. Im not sure if it fires more than once, when new content is loaded , or not.
but for images, you could use something like this.
//Call a function after matching images have finished loading
function imagesLoadedEvent(selector, callback) {
var This = this;
this.images = $(selector);
this.nrImagesToLoad = this.images.length;
this.nrImagesLoaded = 0;
//check if images have already been cached and loaded
$.each(this.images, function (key, img) {
if (img.plete) {
This.nrImagesLoaded++;
}
if (This.nrImagesToLoad == This.nrImagesLoaded) {
callback(This.images);
}
});
this.images.load(function (evt) {
This.nrImagesLoaded++;
if (This.nrImagesToLoad == This.nrImagesLoaded) {
callback(This.images);
}
});
}
$("#btn").click(function () {
var c = $("#container"), evt;
c.empty();
c.append("<img src='" + $("#img1").val() + "' width=94>");
c.append("<img src='" + $("#img2").val() + "' width=94>");
c.append("<img src='" + $("#img3").val() + "' width=94>");
evt = new imagesLoadedEvent("img", allImagesLoaded);
});
function allImagesLoaded(imgs) {
//this is called when all images are loaded
//console.log("all " + imgs.length + " images loaded");
}
js fiddle for images loaded