I am wanting to add an offset from the top and smooth scroll to the following function, The functionality is on one button thats fixed and follows the user down the page. This button has to be able to scroll through numerous anchors and then go back to the first one, with an offset ideally of 105px from the top. trawled the net for hours for help and dont have the jquery know how myself to fix this, any help?? Similar example here - http://www.google.com/nexus/7/ (button in bottom right)
<script>
var max = 6;
function goToNext() {
var hash = String(document.location.hash);
if (hash && hash.indexOf(/anchor/)) {
var newh = Number(hash.replace("#anchor",""));
(newh > max-1) ? newh = 0 : void(null);
document.location.hash = "#anchor" + String(newh+1);
} else {
document.location.hash = "#anchor1";
}
}
</script>
<a href="#" onclick="goToNext();return false;"></a>
<div id="anchor1"></div>
<div id="anchor2"></div>
<div id="anchor3"></div>
<div id="anchor4"></div>
<div id="anchor5"></div>
<div id="anchor6"></div>
You can make it scroll smoothly to the element using
animate({scrollTop:value},delay)
.See this jsfiddle.
Then comes the flickering. Actually it does not flicker but somewhat jerky, if you look closely into the code above. I am setting the
animate(scrollTop)
first, then changing the hashwindow.location.hash = id
. Now when the animate starts scrolling and suddenly we are changing the hash it tends to jump directly to the next<div>
(this is the defaulthaschange
event) but pulled back by theanimate()
and that causes the scrolling to be jerky.We cannot just stop the default propagation of the
haschange
event, there may be a solution to do that but cannot guarantee that it would work on all browsers, each browser has different behaviour when it comes to thehaschange
event. But thanks to @Andy E solution on that SO post you've provided, we don't need to stop the haschange propagation. We can just simply change the hash first, reset it to lastscrollTop()
position then animate scrolling at will!Check this updated jsfiddle.
Now let's talk about HTML5 History API. The reason I didn't introduced this at first because it is implemented differently across HTML5 (especially IE) browsers and has no fallback for HTML4 browsers, making this method somehow inconsistent. But you can get this done properly using a plugin I guess.
Here's how you can do it using
history.pushState()
:See this jsfiddle.
That's it. Cheers!