最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript getBoundingClientRect() values are wrong for dynamically added elements - Stack Overflow

programmeradmin0浏览0评论

Not able to find an answer on wide web so posting the question here.

What I am trying to achieve:

When the user scrolls to the very bottom of the page get the latest dynamically loaded element's bottom value and use it to find out if it is time to load another one.

The math is simple:

if (element.getBoundingClientRect().bottom <= window.innerHeight)
   loadAnotherElement();

window.innerHeight is 955px

The problem:

On initial load the the first element's bottom value is 905px which is fine and trigger the function to load another one, but after the second one is loaded in the bottom value is 1389px which will never trigger the loadAnotherElement function.

I am not able to post full code as it is too plicated so hope the above will be enough to understand.

EDIT

Managed to create a proper test case

Not able to find an answer on wide web so posting the question here.

What I am trying to achieve:

When the user scrolls to the very bottom of the page get the latest dynamically loaded element's bottom value and use it to find out if it is time to load another one.

The math is simple:

if (element.getBoundingClientRect().bottom <= window.innerHeight)
   loadAnotherElement();

window.innerHeight is 955px

The problem:

On initial load the the first element's bottom value is 905px which is fine and trigger the function to load another one, but after the second one is loaded in the bottom value is 1389px which will never trigger the loadAnotherElement function.

I am not able to post full code as it is too plicated so hope the above will be enough to understand.

EDIT

Managed to create a proper test case

Share Improve this question edited Apr 16, 2016 at 10:14 Morpheus asked Apr 15, 2016 at 15:46 MorpheusMorpheus 9,08513 gold badges53 silver badges80 bronze badges 4
  • I am not seeing this issue in a simple test case: jsfiddle/ebmkhjkg/1 - So your issue is probably not javascript doing something unexpected, but maybe your CSS doing something to elements? – somethinghere Commented Apr 15, 2016 at 15:59
  • @somethinghere it took me ages to recreate the problem in jsfiddle but managed in the end. The problem is that elements inside the parent are shifting up to the previous parent. Don't even know how to explain it properly :) please take a look at jsfiddle/ebmkhjkg/2 and you will see the problem – Morpheus Commented Apr 16, 2016 at 10:13
  • I am not seeing the issue, more elements are being added as you hit the bottom. The shifting seems to be more of a CSS issue than anything else... There's occasional holes but it works fine (I am using Safari Technology Preview but i don't think it makes a difference). – somethinghere Commented Apr 16, 2016 at 10:25
  • I don't have safari to test if there are any differences. Tested in Firefox and Chrome, both seem to have the same behavior - when the second batch of elements is loaded the third one doesn't load anymore – Morpheus Commented Apr 16, 2016 at 10:30
Add a ment  | 

1 Answer 1

Reset to default 5

In the JS fiddle you posted the reason it was not recognising the correct height is because your inner elements are floating. A float does not contribute to the height of the parent element, so I will suggest the following solution:

article:after {
    width: 100%;
    clear: both;
    content: '';
    display: block;
}

I have also cleaned up the fiddle a bit an removed the unnecessary parts so it's easier to see where the mistake was (there was no advantage to the table style displayed before and after on your section).

var last = document.querySelector('article');
document.addEventListener('scroll', function(){
    if(last.getBoundingClientRect().bottom <= window.innerHeight){
        var newElement = last.cloneNode(true);
        last.parentNode.appendChild(newElement);
        last = newElement;
    }
});
html,
body {
    height: 100%;
}

/* I have removed a bit of CSS that I think didn't really do anything at all. */

article {
    width: 100%;
    display: inline-block;
    border-bottom: 1px solid red;
}

/* This will create an element that will clear past the floats, stretching your article to the correct encapsulating size */

article:after {
    width: 100%;
    clear: both;
    content: '';
    display: block;
}

/* Some correction so they all stay neatly in line */

div {
    float: left;
    width: 30%;
    margin: 1.5%;
    height: 100vh;
    position: relative;
    background: blue;
}

div:nth-child(3n-2){ background: #666; }
div:nth-child(3n-1){ background: #888;  }
div:nth-child(3n){ background: #222; }
<section>
  <article>
    <!-- I made these 6 divs so they neatly pack six in an article. -->
    <div></div>
    <div></div>
    <div></div>
    <div></div>
    <div></div>
    <div></div>
  </article>
</section>

Update

If you want the elements to neatly stay in line, why bother wrapping them in seperate elements? Simple create the elements as new siblings which will push its final size:

var section = document.querySelector('section');
var article = document.querySelector('article');
document.addEventListener('scroll', function(){
    if(section.getBoundingClientRect().bottom <= window.innerHeight){
        section.appendChild(article.cloneNode(true));
    }
});
html,
body {
    height: 100%;
}

/* I have removed a bit of CSS that I think didn't really do anything at all. */

section {
    width: 100%;
    display: inline-block;
    border-bottom: 1px solid red;
}

/* This will create an element that will clear past the floats, stretching your article to the correct encapsulating size */

section:after {
    width: 100%;
    clear: both;
    content: '';
    display: block;
}

/* Some correction so they all stay neatly in line */

div {
    float: left;
    width: 30%;
    margin: 1.5%;
    height: 100vh;
    position: relative;
    background: blue;
}

div:nth-child(3n-2){ background: #666; }
div:nth-child(3n-1){ background: #888;  }
div:nth-child(3n){ background: #222; }
<section>
  <article>
    <!-- I made these 7 divs for illustrations sake. -->
    <div></div>
    <div></div>
    <div></div>
    <div></div>
    <div></div>
    <div></div>
    <div></div>
  </article>
</section>

发布评论

评论列表(0)

  1. 暂无评论