204 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			204 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								(function($) {
							 | 
						||
| 
								 | 
							
								  var navbarHeight;
							 | 
						||
| 
								 | 
							
								  var initialised = false;
							 | 
						||
| 
								 | 
							
								  var navbarOffset;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  function elOffset($el) {
							 | 
						||
| 
								 | 
							
								    return $el.offset().top - (navbarHeight + navbarOffset);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  function scrollToHash(duringPageLoad) {
							 | 
						||
| 
								 | 
							
								    var elScrollToId = location.hash.replace(/^#/, '');
							 | 
						||
| 
								 | 
							
								    var $el;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    function doScroll() {
							 | 
						||
| 
								 | 
							
								      var offsetTop = elOffset($el);
							 | 
						||
| 
								 | 
							
								      window.scrollTo(window.pageXOffset || window.scrollX, offsetTop);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (elScrollToId) {
							 | 
						||
| 
								 | 
							
								      $el = $(document.getElementById(elScrollToId));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      if (!$el.length) {
							 | 
						||
| 
								 | 
							
								        $el = $(document.getElementsByName(elScrollToId));
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      if ($el.length) {
							 | 
						||
| 
								 | 
							
								        if (duringPageLoad) {
							 | 
						||
| 
								 | 
							
								          $(window).one('scroll', function() {
							 | 
						||
| 
								 | 
							
								            setTimeout(doScroll, 100);
							 | 
						||
| 
								 | 
							
								          });
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								          setTimeout(doScroll, 0);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  function init(opts) {
							 | 
						||
| 
								 | 
							
								    if (initialised) {
							 | 
						||
| 
								 | 
							
								      return;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    initialised = true;
							 | 
						||
| 
								 | 
							
								    navbarHeight = $('.navbar').height();
							 | 
						||
| 
								 | 
							
								    navbarOffset = opts.navbarOffset;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // some browsers move the offset after changing location.
							 | 
						||
| 
								 | 
							
								    // also catch external links coming in
							 | 
						||
| 
								 | 
							
								    $(window).on("hashchange", scrollToHash.bind(null, false));
							 | 
						||
| 
								 | 
							
								    $(scrollToHash.bind(null, true));
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  $.catchAnchorLinks = function(options) {
							 | 
						||
| 
								 | 
							
								    var opts = $.extend({}, jQuery.fn.toc.defaults, options);
							 | 
						||
| 
								 | 
							
								    init(opts);
							 | 
						||
| 
								 | 
							
								  };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  $.fn.toc = function(options) {
							 | 
						||
| 
								 | 
							
								    var self = this;
							 | 
						||
| 
								 | 
							
								    var opts = $.extend({}, jQuery.fn.toc.defaults, options);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    var container = $(opts.container);
							 | 
						||
| 
								 | 
							
								    var tocs = [];
							 | 
						||
| 
								 | 
							
								    var headings = $(opts.selectors, container);
							 | 
						||
| 
								 | 
							
								    var headingOffsets = [];
							 | 
						||
| 
								 | 
							
								    var activeClassName = 'active';
							 | 
						||
| 
								 | 
							
								    var ANCHOR_PREFIX = "__anchor";
							 | 
						||
| 
								 | 
							
								    var maxScrollTo;
							 | 
						||
| 
								 | 
							
								    var visibleHeight;
							 | 
						||
| 
								 | 
							
								    var headerHeight = 10; // so if the header is readable, its counted as shown
							 | 
						||
| 
								 | 
							
								    init();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    var scrollTo = function(e) {
							 | 
						||
| 
								 | 
							
								      e.preventDefault();
							 | 
						||
| 
								 | 
							
								      var target = $(e.target);
							 | 
						||
| 
								 | 
							
								      if (target.prop('tagName').toLowerCase() !== "a") {
							 | 
						||
| 
								 | 
							
								        target = target.parent();
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								      var elScrollToId = target.attr('href').replace(/^#/, '') + ANCHOR_PREFIX;
							 | 
						||
| 
								 | 
							
								      var $el = $(document.getElementById(elScrollToId));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      var offsetTop = Math.min(maxScrollTo, elOffset($el));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      $('body,html').animate({ scrollTop: offsetTop }, 400, 'swing', function() {
							 | 
						||
| 
								 | 
							
								        location.hash = '#' + elScrollToId;
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      $('a', self).removeClass(activeClassName);
							 | 
						||
| 
								 | 
							
								      target.addClass(activeClassName);
							 | 
						||
| 
								 | 
							
								    };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    var calcHadingOffsets = function() {
							 | 
						||
| 
								 | 
							
								      maxScrollTo = $("body").height() - $(window).height();
							 | 
						||
| 
								 | 
							
								      visibleHeight = $(window).height() - navbarHeight;
							 | 
						||
| 
								 | 
							
								      headingOffsets = [];
							 | 
						||
| 
								 | 
							
								      headings.each(function(i, heading) {
							 | 
						||
| 
								 | 
							
								        var anchorSpan = $(heading).prev("span");
							 | 
						||
| 
								 | 
							
								        var top = 0;
							 | 
						||
| 
								 | 
							
								        if (anchorSpan.length) {
							 | 
						||
| 
								 | 
							
								          top = elOffset(anchorSpan);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        headingOffsets.push(top > 0 ? top : 0);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    //highlight on scroll
							 | 
						||
| 
								 | 
							
								    var timeout;
							 | 
						||
| 
								 | 
							
								    var highlightOnScroll = function(e) {
							 | 
						||
| 
								 | 
							
								      if (!tocs.length) {
							 | 
						||
| 
								 | 
							
								        return;
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								      if (timeout) {
							 | 
						||
| 
								 | 
							
								        clearTimeout(timeout);
							 | 
						||
| 
								 | 
							
								      }
							 | 
						||
| 
								 | 
							
								      timeout = setTimeout(function() {
							 | 
						||
| 
								 | 
							
								        var top = $(window).scrollTop(),
							 | 
						||
| 
								 | 
							
								          highlighted;
							 | 
						||
| 
								 | 
							
								        for (var i = headingOffsets.length - 1; i >= 0; i--) {
							 | 
						||
| 
								 | 
							
								          var isActive = tocs[i].hasClass(activeClassName);
							 | 
						||
| 
								 | 
							
								          // at the end of the page, allow any shown header
							 | 
						||
| 
								 | 
							
								          if (isActive && headingOffsets[i] >= maxScrollTo && top >= maxScrollTo) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								          }
							 | 
						||
| 
								 | 
							
								          // if we have got to the first heading or the heading is the first one visible
							 | 
						||
| 
								 | 
							
								          if (i === 0 || (headingOffsets[i] + headerHeight >= top && (headingOffsets[i - 1] + headerHeight <= top))) {
							 | 
						||
| 
								 | 
							
								            // in the case that a heading takes up more than the visible height e.g. we are showing
							 | 
						||
| 
								 | 
							
								            // only the one above, highlight the one above
							 | 
						||
| 
								 | 
							
								            if (i > 0 && headingOffsets[i] - visibleHeight >= top) {
							 | 
						||
| 
								 | 
							
								              i--;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $('a', self).removeClass(activeClassName);
							 | 
						||
| 
								 | 
							
								            if (i >= 0) {
							 | 
						||
| 
								 | 
							
								              highlighted = tocs[i].addClass(activeClassName);
							 | 
						||
| 
								 | 
							
								              opts.onHighlight(highlighted);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            break;
							 | 
						||
| 
								 | 
							
								          }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								      }, 50);
							 | 
						||
| 
								 | 
							
								    };
							 | 
						||
| 
								 | 
							
								    if (opts.highlightOnScroll) {
							 | 
						||
| 
								 | 
							
								      $(window).bind('scroll', highlightOnScroll);
							 | 
						||
| 
								 | 
							
								      $(window).bind('load resize', function() {
							 | 
						||
| 
								 | 
							
								        calcHadingOffsets();
							 | 
						||
| 
								 | 
							
								        highlightOnScroll();
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return this.each(function() {
							 | 
						||
| 
								 | 
							
								      //build TOC
							 | 
						||
| 
								 | 
							
								      var el = $(this);
							 | 
						||
| 
								 | 
							
								      var ul = $('<div class="list-group">');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      headings.each(function(i, heading) {
							 | 
						||
| 
								 | 
							
								        var $h = $(heading);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        var anchor = $('<span/>').attr('id', opts.anchorName(i, heading, opts.prefix) + ANCHOR_PREFIX).insertBefore($h);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        var span = $('<span/>')
							 | 
						||
| 
								 | 
							
								          .text(opts.headerText(i, heading, $h));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        //build TOC item
							 | 
						||
| 
								 | 
							
								        var a = $('<a class="list-group-item"/>')
							 | 
						||
| 
								 | 
							
								          .append(span)
							 | 
						||
| 
								 | 
							
								          .attr('href', '#' + opts.anchorName(i, heading, opts.prefix))
							 | 
						||
| 
								 | 
							
								          .bind('click', function(e) {
							 | 
						||
| 
								 | 
							
								            scrollTo(e);
							 | 
						||
| 
								 | 
							
								            el.trigger('selected', $(this).attr('href'));
							 | 
						||
| 
								 | 
							
								          });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        span.addClass(opts.itemClass(i, heading, $h, opts.prefix));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        tocs.push(a);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        ul.append(a);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								      el.html(ul);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      calcHadingOffsets();
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								jQuery.fn.toc.defaults = {
							 | 
						||
| 
								 | 
							
								  container: 'body',
							 | 
						||
| 
								 | 
							
								  selectors: 'h1,h2,h3',
							 | 
						||
| 
								 | 
							
								  smoothScrolling: true,
							 | 
						||
| 
								 | 
							
								  prefix: 'toc',
							 | 
						||
| 
								 | 
							
								  onHighlight: function() {},
							 | 
						||
| 
								 | 
							
								  highlightOnScroll: true,
							 | 
						||
| 
								 | 
							
								  navbarOffset: 0,
							 | 
						||
| 
								 | 
							
								  anchorName: function(i, heading, prefix) {
							 | 
						||
| 
								 | 
							
								    return prefix+i;
							 | 
						||
| 
								 | 
							
								  },
							 | 
						||
| 
								 | 
							
								  headerText: function(i, heading, $heading) {
							 | 
						||
| 
								 | 
							
								    return $heading.text();
							 | 
						||
| 
								 | 
							
								  },
							 | 
						||
| 
								 | 
							
								  itemClass: function(i, heading, $heading, prefix) {
							 | 
						||
| 
								 | 
							
								    return prefix + '-' + $heading[0].tagName.toLowerCase();
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								})(jQuery);
							 |