Учет фиксированного заголовка с помощью animate.scrolltop и (target) .offset (). Top; - PullRequest
13 голосов
/ 30 января 2012

Это должен быть довольно простой вопрос, но я потратил большую часть своего утра на это, и в этот момент я близок к тому, чтобы бросить полотенце.У меня нет даже немного 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;

 });
});

Я пытаюсь получитьприземлиться на 30px выше смещения (). top - я попытался

$('html, body').stop().animate({ scrollTop: scrollToPosition -30}, 600,

Что почти работает - он идет в нужное место, но затем отскакивает назад.

Я также пытался

scrollTop: $(target).offset().top - 20 },

Я также пытался

scrollTop: $(hash).offset().top + $('#access').outerHeight()

Что, кажется, ничего не меняет,

Кажется, что ответ может быть здесь: Проблема прокрутки страницы JQuery с фиксированным заголовком , но я просто не могу понять, как это получить.

Я знаю, что это похоже на другие вопросы - но я прошел через то, что смог найти, и я настолько неграмотен, что не смог скопировать / вставить что-либо, что решает проблему.

Я был бы невероятно благодарен за решение.

Большое спасибо,

Martin

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;
        }
    }
  });
});

Ответы [ 3 ]

16 голосов
/ 30 января 2012

РЕДАКТИРОВАНИЕ: Вам просто нужно определить высоту фиксированного заголовка и вычесть его из 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 . С этим вы можете использовать ваши события hashchange гораздо больше. Вот пример использования:

// 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's 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 Address обнаружит ссылку и автоматически запустит событие изменения хеша.

7 голосов
/ 03 января 2013

Я знаю, что это старый вопрос (вид), но я столкнулся с аналогичной проблемой с фиксированной выпадающей навигацией на веб-сайте.Обратите внимание, что это фрагмент кода с плавной прокруткой, хотя вы можете легко сделать его автоматическим, изменив скорость анимации.

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 голосов
/ 19 февраля 2014

Я подправил оригинальный скрипт из 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 ;
        });
    });
});

Да, я немного опоздал, но эта проблема только что возникла у меня ... Ура! * * 1006

...