01 Jun 2018

This article will take about 2 minutes to read.
No comments

This is a weird one, and not something you will stumble into every day. But it is a bug nonetheless.

On iOS safari, if you use window.scrollTo(0, y) and y is larger than document’s maximum scroll, any immediate call to getBoundingClientRect will return Same will happen for horizontal scroll and left value.

What happens is that browser thinks it actually scrolled to y and calculates element’s position based on that scroll position. This happens only if scrollTo and getBoundingClientRect are executed one right after the other.

Not even requestAnimationFrame will save you. Adding a small timeout will, but that is not a viable solution.

Check the demo Unfortunately I couldn’t include the iframe with the demo, because of the other bug. Today’s your lucky day, you got two bugs by the price of one!

I’ve tested it only on iOS 11, but I guess other versions are affected as well.

Solution

Solution is easy, we need to determine maximum possible scroll and to cap our y value.

Helper method:

function getPageMaxScroll() {
  // Cross browser page height detection is ugly
  return Math.max(
    document.body.scrollHeight,
    document.body.offsetHeight,
    document.documentElement.clientHeight,
    document.documentElement.scrollHeight,
    document.documentElement.offsetHeight
  ) - window.innerHeight; // Subtract viewport height
}

Usage:

let top = 1000000; // Value larger than maximum scroll
const maxScroll = getPageMaxScroll();

// Fix for bug on iOS devices
// When top was larger than maximum page scroll
// "getBoundingClientRect" would take that value into calculations
if (top > maxScroll) {
  top = maxScroll;
}

// Scroll the window to the new position
window.scrollTo(0, top);

// Get the new position
const rect = this.contentWrapperElement.getBoundingClientRect();

Hope that helps!

P.S. There is another bug on iOS Safari with getBoundingClientRect and position: fixed, but that one is documented and tracked.

Category
JavaScript

No comments, be the first to comment

Sending failed, please try again.
Thank you! Your comment is sent. Please note that all of the comments go through moderation.