toc.js 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. (function($) {
  2. var navbarHeight;
  3. var initialised = false;
  4. var navbarOffset;
  5. function elOffset($el) {
  6. return $el.offset().top - (navbarHeight + navbarOffset);
  7. }
  8. function scrollToHash(duringPageLoad) {
  9. var elScrollToId = location.hash.replace(/^#/, '');
  10. var $el;
  11. function doScroll() {
  12. var offsetTop = elOffset($el);
  13. window.scrollTo(window.pageXOffset || window.scrollX, offsetTop);
  14. }
  15. if (elScrollToId) {
  16. $el = $(document.getElementById(elScrollToId));
  17. if (!$el.length) {
  18. $el = $(document.getElementsByName(elScrollToId));
  19. }
  20. if ($el.length) {
  21. if (duringPageLoad) {
  22. $(window).one('scroll', function() {
  23. setTimeout(doScroll, 100);
  24. });
  25. } else {
  26. setTimeout(doScroll, 0);
  27. }
  28. }
  29. }
  30. }
  31. function init(opts) {
  32. if (initialised) {
  33. return;
  34. }
  35. initialised = true;
  36. navbarHeight = $('.navbar').height();
  37. navbarOffset = opts.navbarOffset;
  38. // some browsers move the offset after changing location.
  39. // also catch external links coming in
  40. $(window).on("hashchange", scrollToHash.bind(null, false));
  41. $(scrollToHash.bind(null, true));
  42. }
  43. $.catchAnchorLinks = function(options) {
  44. var opts = $.extend({}, jQuery.fn.toc.defaults, options);
  45. init(opts);
  46. };
  47. $.fn.toc = function(options) {
  48. var self = this;
  49. var opts = $.extend({}, jQuery.fn.toc.defaults, options);
  50. var container = $(opts.container);
  51. var tocs = [];
  52. var headings = $(opts.selectors, container);
  53. var headingOffsets = [];
  54. var activeClassName = 'active';
  55. var ANCHOR_PREFIX = "__anchor";
  56. var maxScrollTo;
  57. var visibleHeight;
  58. var headerHeight = 10; // so if the header is readable, its counted as shown
  59. init();
  60. var scrollTo = function(e) {
  61. e.preventDefault();
  62. var target = $(e.target);
  63. if (target.prop('tagName').toLowerCase() !== "a") {
  64. target = target.parent();
  65. }
  66. var elScrollToId = target.attr('href').replace(/^#/, '') + ANCHOR_PREFIX;
  67. var $el = $(document.getElementById(elScrollToId));
  68. var offsetTop = Math.min(maxScrollTo, elOffset($el));
  69. $('body,html').animate({ scrollTop: offsetTop }, 400, 'swing', function() {
  70. location.hash = '#' + elScrollToId;
  71. });
  72. $('a', self).removeClass(activeClassName);
  73. target.addClass(activeClassName);
  74. };
  75. var calcHadingOffsets = function() {
  76. maxScrollTo = $("body").height() - $(window).height();
  77. visibleHeight = $(window).height() - navbarHeight;
  78. headingOffsets = [];
  79. headings.each(function(i, heading) {
  80. var anchorSpan = $(heading).prev("span");
  81. var top = 0;
  82. if (anchorSpan.length) {
  83. top = elOffset(anchorSpan);
  84. }
  85. headingOffsets.push(top > 0 ? top : 0);
  86. });
  87. }
  88. //highlight on scroll
  89. var timeout;
  90. var highlightOnScroll = function(e) {
  91. if (!tocs.length) {
  92. return;
  93. }
  94. if (timeout) {
  95. clearTimeout(timeout);
  96. }
  97. timeout = setTimeout(function() {
  98. var top = $(window).scrollTop(),
  99. highlighted;
  100. for (var i = headingOffsets.length - 1; i >= 0; i--) {
  101. var isActive = tocs[i].hasClass(activeClassName);
  102. // at the end of the page, allow any shown header
  103. if (isActive && headingOffsets[i] >= maxScrollTo && top >= maxScrollTo) {
  104. return;
  105. }
  106. // if we have got to the first heading or the heading is the first one visible
  107. if (i === 0 || (headingOffsets[i] + headerHeight >= top && (headingOffsets[i - 1] + headerHeight <= top))) {
  108. // in the case that a heading takes up more than the visible height e.g. we are showing
  109. // only the one above, highlight the one above
  110. if (i > 0 && headingOffsets[i] - visibleHeight >= top) {
  111. i--;
  112. }
  113. $('a', self).removeClass(activeClassName);
  114. if (i >= 0) {
  115. highlighted = tocs[i].addClass(activeClassName);
  116. opts.onHighlight(highlighted);
  117. }
  118. break;
  119. }
  120. }
  121. }, 50);
  122. };
  123. if (opts.highlightOnScroll) {
  124. $(window).bind('scroll', highlightOnScroll);
  125. $(window).bind('load resize', function() {
  126. calcHadingOffsets();
  127. highlightOnScroll();
  128. });
  129. }
  130. return this.each(function() {
  131. //build TOC
  132. var el = $(this);
  133. var ul = $('<div class="list-group">');
  134. headings.each(function(i, heading) {
  135. var $h = $(heading);
  136. var anchor = $('<span/>').attr('id', opts.anchorName(i, heading, opts.prefix) + ANCHOR_PREFIX).insertBefore($h);
  137. var span = $('<span/>')
  138. .text(opts.headerText(i, heading, $h));
  139. //build TOC item
  140. var a = $('<a class="list-group-item"/>')
  141. .append(span)
  142. .attr('href', '#' + opts.anchorName(i, heading, opts.prefix))
  143. .bind('click', function(e) {
  144. scrollTo(e);
  145. el.trigger('selected', $(this).attr('href'));
  146. });
  147. span.addClass(opts.itemClass(i, heading, $h, opts.prefix));
  148. tocs.push(a);
  149. ul.append(a);
  150. });
  151. el.html(ul);
  152. calcHadingOffsets();
  153. });
  154. };
  155. jQuery.fn.toc.defaults = {
  156. container: 'body',
  157. selectors: 'h1,h2,h3',
  158. smoothScrolling: true,
  159. prefix: 'toc',
  160. onHighlight: function() {},
  161. highlightOnScroll: true,
  162. navbarOffset: 0,
  163. anchorName: function(i, heading, prefix) {
  164. return prefix+i;
  165. },
  166. headerText: function(i, heading, $heading) {
  167. return $heading.text();
  168. },
  169. itemClass: function(i, heading, $heading, prefix) {
  170. return prefix + '-' + $heading[0].tagName.toLowerCase();
  171. }
  172. };
  173. })(jQuery);