bootstrap-scrollspy.js 4.5 KB
Newer Older
Jacob Thornton's avatar
Jacob Thornton committed
1
/* =============================================================
Mark Otto's avatar
Mark Otto committed
2
 * bootstrap-scrollspy.js v2.2.2
Jacob Thornton's avatar
Jacob Thornton committed
3
4
 * http://twitter.github.com/bootstrap/javascript.html#scrollspy
 * =============================================================
Mark Otto's avatar
Mark Otto committed
5
 * Copyright 2012 Twitter, Inc.
Jacob Thornton's avatar
Jacob Thornton committed
6
7
8
9
10
11
12
13
14
15
16
17
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
Jacob Thornton's avatar
Jacob Thornton committed
18
19
 * ============================================================== */

20

21
22
23
!function ($) {

  "use strict"; // jshint ;_;
Jacob Thornton's avatar
Jacob Thornton committed
24

25

Jacob Thornton's avatar
Jacob Thornton committed
26
27
 /* SCROLLSPY CLASS DEFINITION
  * ========================== */
28

Jacob Thornton's avatar
Jacob Thornton committed
29
  function ScrollSpy(element, options) {
30
    var process = $.proxy(this.process, this)
31
      , $element = $(element).is('body') ? $(window) : $(element)
Jacob Thornton's avatar
Jacob Thornton committed
32
      , href
33
    this.options = $.extend({}, $.fn.scrollspy.defaults, options)
Jacob Thornton's avatar
Jacob Thornton committed
34
    this.$scrollElement = $element.on('scroll.scroll-spy.data-api', process)
35
    this.selector = (this.options.target
Jacob Thornton's avatar
Jacob Thornton committed
36
      || ((href = $(element).attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
37
      || '') + ' .nav li > a'
Jacob Thornton's avatar
Jacob Thornton committed
38
    this.$body = $('body')
39
    this.refresh()
40
    this.process()
Jacob Thornton's avatar
Jacob Thornton committed
41
42
43
44
  }

  ScrollSpy.prototype = {

45
46
47
      constructor: ScrollSpy

    , refresh: function () {
48
49
50
        var self = this
          , $targets

51
52
        this.offsets = $([])
        this.targets = $([])
53
54

        $targets = this.$body
55
56
          .find(this.selector)
          .map(function () {
57
58
            var $el = $(this)
              , href = $el.data('target') || $el.attr('href')
Jacob Thornton's avatar
Jacob Thornton committed
59
              , $href = /^#\w/.test(href) && $(href)
Jacob Thornton's avatar
Jacob Thornton committed
60
            return ( $href
Jacob Thornton's avatar
Jacob Thornton committed
61
              && $href.length
Godric's avatar
Godric committed
62
              && [[ $href.position().top + self.$scrollElement.scrollTop(), href ]] ) || null
63
64
65
66
67
          })
          .sort(function (a, b) { return a[0] - b[0] })
          .each(function () {
            self.offsets.push(this[0])
            self.targets.push(this[1])
68
          })
Jacob Thornton's avatar
Jacob Thornton committed
69
70
      }

71
    , process: function () {
72
        var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
73
74
          , scrollHeight = this.$scrollElement[0].scrollHeight || this.$body[0].scrollHeight
          , maxScroll = scrollHeight - this.$scrollElement.height()
Jacob Thornton's avatar
Jacob Thornton committed
75
76
77
78
79
          , offsets = this.offsets
          , targets = this.targets
          , activeTarget = this.activeTarget
          , i

80
81
        if (scrollTop >= maxScroll) {
          return activeTarget != (i = targets.last()[0])
Jacob Thornton's avatar
;)    
Jacob Thornton committed
82
            && this.activate ( i )
83
84
        }

Jacob Thornton's avatar
Jacob Thornton committed
85
86
87
88
        for (i = offsets.length; i--;) {
          activeTarget != targets[i]
            && scrollTop >= offsets[i]
            && (!offsets[i + 1] || scrollTop <= offsets[i + 1])
89
            && this.activate( targets[i] )
Jacob Thornton's avatar
Jacob Thornton committed
90
91
92
        }
      }

93
94
    , activate: function (target) {
        var active
95
          , selector
96

Jacob Thornton's avatar
Jacob Thornton committed
97
98
        this.activeTarget = target

Jacob Thornton's avatar
Jacob Thornton committed
99
100
        $(this.selector)
          .parent('.active')
Jacob Thornton's avatar
Jacob Thornton committed
101
102
          .removeClass('active')

103
104
105
106
107
        selector = this.selector
          + '[data-target="' + target + '"],'
          + this.selector + '[href="' + target + '"]'

        active = $(selector)
Jacob Thornton's avatar
Jacob Thornton committed
108
109
          .parent('li')
          .addClass('active')
110

111
        if (active.parent('.dropdown-menu').length)  {
Jacob Thornton's avatar
Jacob Thornton committed
112
          active = active.closest('li.dropdown').addClass('active')
113
        }
Jacob Thornton's avatar
Jacob Thornton committed
114

Jacob Thornton's avatar
Jacob Thornton committed
115
        active.trigger('activate')
Jacob Thornton's avatar
Jacob Thornton committed
116
117
118
119
      }

  }

120
121
122
123

 /* SCROLLSPY PLUGIN DEFINITION
  * =========================== */

124
125
  var old = $.fn.scrollspy

Jacob Thornton's avatar
Jacob Thornton committed
126
  $.fn.scrollspy = function (option) {
127
128
129
    return this.each(function () {
      var $this = $(this)
        , data = $this.data('scrollspy')
130
131
        , options = typeof option == 'object' && option
      if (!data) $this.data('scrollspy', (data = new ScrollSpy(this, options)))
132
133
134
135
      if (typeof option == 'string') data[option]()
    })
  }

136
  $.fn.scrollspy.Constructor = ScrollSpy
137

138
139
140
141
  $.fn.scrollspy.defaults = {
    offset: 10
  }

142

143
144
145
146
147
148
149
150
151
 /* SCROLLSPY NO CONFLICT
  * ===================== */

  $.fn.scrollspy.noConflict = function () {
    $.fn.scrollspy = old
    return this
  }


152
 /* SCROLLSPY DATA-API
153
  * ================== */
154

Jacob Thornton's avatar
Jacob Thornton committed
155
  $(window).on('load', function () {
156
157
158
159
    $('[data-spy="scroll"]').each(function () {
      var $spy = $(this)
      $spy.scrollspy($spy.data())
    })
160
  })
161

162
}(window.jQuery);