Учет фиксированного заголовка с анимацией. Scrolltop и (target).offset(). Top;
Это должен быть довольно простой вопрос, но я бросил большую часть своего утра на это, и в этот момент я близок к тому, чтобы бросить полотенце. У меня даже немного js foo, но я нашел красиво прокомментированный фрагмент кода, который я надеюсь использовать для анимации якорных ссылок:
$(document).ready(function() {
$('a[href*=#]').bind('click', function(e) {
e.preventDefault(); //prevent the "normal" behaviour which would be a "hard" jump
var target = $(this).attr("href"); //Get the target
var scrollToPosition = $(target).offset().top;
// perform animated scrolling by getting top-position of target-element and set it as scroll target
$('html, body').stop().animate({ scrollTop: scrollToPosition}, 600, function() {
location.hash = target; //attach the hash (#jumptarget) to the pageurl
});
return false;
});
});
Я пытаюсь заставить его приземлиться на 30 пикселей выше смещения(). top - Я пробовал
$('html, body').stop().animate({ scrollTop: scrollToPosition -30}, 600,
Что почти работает - он идет в нужное место, но затем отскакивает назад.
Я также пробовал
scrollTop: $(target).offset().top - 20 },
Я также пробовал
scrollTop: $(hash).offset().top + $('#access').outerHeight()
Что, кажется, ничего не меняет.
Кажется, что ответ может быть здесь: Проблема с прокруткой страницы JQuery с фиксированным заголовком, но я просто не могу ее получить.
Я знаю, что это похоже на другие вопросы, но я прошел через то, что мог найти, и я недостаточно грамотен, что мне не удалось скопировать/вставить все, что исправляет проблему.
Я был бы невероятно благодарен за решение.
Большое спасибо,
Мартин
PS
Этот другой фрагмент кода, который я нашел, работает, но он лишает хэштегов, что делает его в основном бесполезным.
$(function(){
$('a[href*=#]').click(function() {
if (location.pathname.replace(/^\//,'') == this.pathname.replace(/^\//,'')
&& location.hostname == this.hostname) {
var $target = $(this.hash);
$target = $target.length && $target || $('[name=' + this.hash.slice(1) +']');
if ($target.length) {
var targetOffset = $target.offset().top;
$('html,body').animate({scrollTop: targetOffset - 30}, 1000);
return false;
}
}
});
});
Ответы
Ответ 1
Редакция:
Вам просто нужно определить высоту фиксированного заголовка и вычесть из scrollToPosition
, который вы делали правильно. Проблема заключается в том, что window.location.hash = "" + target;
перескакивает страницу в начало элемента с этим идентификатором. Поэтому, если вы живите там, как вы делали, а затем переходите на этот хеш, он будет "возвращаться", как вы описали. Вот первый способ борьбы с этим:
// Get the height of the header
var headerHeight = $("div#header").height();
// Attach the click event
$('a[href*=#]').bind("click", function(e) {
e.preventDefault();
var target = $(this).attr("href"); //Get the target
var scrollToPosition = $(target).offset().top - headerHeight;
$('html').animate({ 'scrollTop': scrollToPosition }, 600, function(){
window.location.hash = "" + target;
// This hash change will jump the page to the top of the div with the same id
// so we need to force the page to back to the end of the animation
$('html').animate({ 'scrollTop': scrollToPosition }, 0);
});
$('body').append("called");
});
Здесь новый jsfiddle для этого первого метода: http://jsfiddle.net/yjcRv/1/
ДАЛЬНЕЙШЕЕ ИЗДАНИЕ:
Еще лучший способ управления событиями изменения хэша - использовать плагин, например jQuery Address. С этим вы можете использовать свои события хэшмена намного больше. Вот пример использования:
// Get the height of the header
var headerHeight = $("div#header").height();
$.address.change(function(evt){
var target = "#" + evt["pathNames"][0]; //Get the target from the event data
// If there been some content requested go to it…else go to the top
if(evt["pathNames"][0]){
var scrollToPosition = $(target).offset().top - headerHeight;
$('html').animate({ 'scrollTop': scrollToPosition }, 600);
}else{
$('html').animate({ 'scrollTop': '0' }, 600);
}
return false;
});
// Attach the click event
$('a').bind("click", function(e) {
// Change the location
$.address.value($(this).attr("href"));
return false;
});
Здесь приведен пример: http://www.vdotgood.com/stack/user3444.html
ПРИМЕЧАНИЕ. Теперь вам не нужно добавлять хеш к атрибуту href ссылок. Здесь ссылка, с которой вы можете настроить селектор jQuery:
<!-- This is correct -->
<a href="/target" class="myclass">Target</a>
<!-- These are incorrect -->
<a href="/#/target" class="myclass">Target</a>
<a href="#/target" class="myclass">Target</a>
Чтобы настроить таргетинг на эту ссылку, используйте селектор, например:
$("a.myclass").click(function(){
$.address.value($(this).attr("href"));
return false;
});
jQuery Address действительно ищет ссылки, которые имеют следующий атрибут:
<a href="/target" rel="address:/target">Target</a>
Атрибут rel
содержит address:
, за которым следует относительный url, определенный вами в этом случае /target
. Если вы используете это, адрес jQuery будет обнаруживать связь и автоматически запускать событие изменения хэша.
Ответ 2
Я знаю, что это старый вопрос (вроде), но у меня возникла аналогичная проблема с фиксированной навигацией по выпадающему меню на веб-сайте. Обратите внимание, что это гладкий фрагмент кода прокрутки, хотя вы можете легко сделать это автоматически, изменив скорость анимации.
JQuery
$('body').on('click','a[href^="#"]',function(event){
event.preventDefault();
var target_offset = $(this.hash).offset() ? $(this.hash).offset().top : 0;
//change this number to create the additional off set
var customoffset = 75
$('html, body').animate({scrollTop:target_offset - customoffset}, 500);
});
Я использовал этот кусок кода в течение длительного времени без каких-либо проблем. Единственное, что мне не нравится в этом, это захват ЛЮБОГО тега. Таким образом, в плагине, таком как плагин Flexslider, где используется навигационная система, я вручную удаляю их из плагина.
Ответ 3
Я изменил оригинальный script из http://www.paulund.co.uk/smooth-scroll-to-internal-links-with-jquery. Это чудеса, но вы не можете установить задержку, как есть.
var headerHeight = $("header").height();
$(document).ready(function(){
$('a[href^="#"]').on('click',function (e) {
e.preventDefault();
var target = this.hash,
$target = $(target);
$('html, body').stop().animate({
'scrollTop': $target.offset().top - headerHeight
}, 1200, 'swing', function () {
window.location.hash = target ;
});
});
});
Да, я немного опоздал, но эта проблема возникла только мне...
Ура!