Safari in ios8 is scrolling screen when fixed elements get focus Safari in ios8 is scrolling screen when fixed elements get focus ios ios

Safari in ios8 is scrolling screen when fixed elements get focus


Based on this good analysis of this issue, I've used this in html and body elements in css:

html,body{    -webkit-overflow-scrolling : touch !important;    overflow: auto !important;    height: 100% !important;}

I think it's working great for me.


The best solution I could come up with is to switch to using position: absolute; on focus and calculating the position it was at when it was using position: fixed;. The trick is that the focus event fires too late, so touchstart must be used.

The solution in this answer mimics the correct behavior we had in iOS 7 very closely.

Requirements:

The body element must have positioning in order to ensure proper positioning when the element switches to absolute positioning.

body {    position: relative;}

The Code (Live Example):

The following code is a basic example for the provided test-case, and can be adapted for your specific use-case.

//Get the fixed element, and the input element it contains.var fixed_el = document.getElementById('b');var input_el = document.querySelector('textarea');//Listen for touchstart, focus will fire too late.input_el.addEventListener('touchstart', function() {    //If using a non-px value, you will have to get clever, or just use 0 and live with the temporary jump.    var bottom = parseFloat(window.getComputedStyle(fixed_el).bottom);    //Switch to position absolute.    fixed_el.style.position = 'absolute';    fixed_el.style.bottom = (document.height - (window.scrollY + window.innerHeight) + bottom) + 'px';    //Switch back when focus is lost.    function blured() {        fixed_el.style.position = '';        fixed_el.style.bottom = '';        input_el.removeEventListener('blur', blured);    }    input_el.addEventListener('blur', blured);});

Here is the same code without the hack for comparison.

Caveat:

If the position: fixed; element has any other parent elements with positioning besides body, switching to position: absolute; may have unexpected behavior. Due to the nature of position: fixed; this is probably not a major issue, since nesting such elements is not common.

Recommendations:

While the use of the touchstart event will filter out most desktop environments, you will probably want to use user-agent sniffing so that this code will only run for the broken iOS 8, and not other devices such as Android and older iOS versions. Unfortunately, we don't yet know when Apple will fix this issue in iOS, but I would be surprised if it is not fixed in the next major version.


I found a method that works without the need to change to position absolute!

Full uncommented code

var scrollPos = $(document).scrollTop();$(window).scroll(function(){    scrollPos = $(document).scrollTop();});var savedScrollPos = scrollPos;function is_iOS() {  var iDevices = [    'iPad Simulator',    'iPhone Simulator',    'iPod Simulator',    'iPad',    'iPhone',    'iPod'  ];  while (iDevices.length) {    if (navigator.platform === iDevices.pop()){ return true; }  }  return false;}$('input[type=text]').on('touchstart', function(){    if (is_iOS()){        savedScrollPos = scrollPos;        $('body').css({            position: 'relative',            top: -scrollPos        });        $('html').css('overflow','hidden');    }}).blur(function(){    if (is_iOS()){        $('body, html').removeAttr('style');        $(document).scrollTop(savedScrollPos);    }});

Breaking it down

First you need to have the fixed input field toward the top of the page in the HTML (it's a fixed element so it should semantically make sense to have it near the top anyway):

<!DOCTYPE HTML><html>    <head>      <title>Untitled</title>    </head>    <body>        <form class="fixed-element">            <input class="thing-causing-the-issue" type="text" />        </form>        <div class="everything-else">(content)</div>    </body></html>

Then you need to save the current scroll position into global variables:

//Always know the current scroll positionvar scrollPos = $(document).scrollTop();$(window).scroll(function(){    scrollPos = $(document).scrollTop();});//need to be able to save current scroll pos while keeping actual scroll pos up to datevar savedScrollPos = scrollPos;

Then you need a way to detect iOS devices so it doesn't affect things that don't need the fix (function taken from https://stackoverflow.com/a/9039885/1611058)

//function for testing if it is an iOS devicefunction is_iOS() {  var iDevices = [    'iPad Simulator',    'iPhone Simulator',    'iPod Simulator',    'iPad',    'iPhone',    'iPod'  ];  while (iDevices.length) {    if (navigator.platform === iDevices.pop()){ return true; }  }  return false;}

Now that we have everything we need, here is the fix :)

//when user touches the input$('input[type=text]').on('touchstart', function(){    //only fire code if it's an iOS device    if (is_iOS()){        //set savedScrollPos to the current scroll position        savedScrollPos = scrollPos;        //shift the body up a number of pixels equal to the current scroll position        $('body').css({            position: 'relative',            top: -scrollPos        });        //Hide all content outside of the top of the visible area        //this essentially chops off the body at the position you are scrolled to so the browser can't scroll up any higher        $('html').css('overflow','hidden');    }})//when the user is done and removes focus from the input field.blur(function(){    //checks if it is an iOS device    if (is_iOS()){        //Removes the custom styling from the body and html attribute        $('body, html').removeAttr('style');        //instantly scrolls the page back down to where you were when you clicked on input field        $(document).scrollTop(savedScrollPos);    }});