bootstrap-dropdown.js 4.16 KB
Newer Older
Jacob Thornton's avatar
Jacob Thornton committed
1
/* ============================================================
Mark Otto's avatar
Mark Otto committed
2
 * bootstrap-dropdown.js v2.2.3
Jacob Thornton's avatar
Jacob Thornton committed
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
 * http://twitter.github.com/bootstrap/javascript.html#dropdowns
 * ============================================================
 * Copyright 2012 Twitter, Inc.
 *
 * 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.
 * ============================================================ */


21
22
23
!function ($) {

  "use strict"; // jshint ;_;
Jacob Thornton's avatar
Jacob Thornton committed
24
25
26
27
28


 /* DROPDOWN CLASS DEFINITION
  * ========================= */

29
  var toggle = '[data-toggle=dropdown]'
30
    , Dropdown = function (element) {
Jacob Thornton's avatar
Jacob Thornton committed
31
32
33
34
35
36
37
38
39
40
        var $el = $(element).on('click.dropdown.data-api', this.toggle)
        $('html').on('click.dropdown.data-api', function () {
          $el.parent().removeClass('open')
        })
      }

  Dropdown.prototype = {

    constructor: Dropdown

41
  , toggle: function (e) {
Jacob Thornton's avatar
Jacob Thornton committed
42
43
44
45
      var $this = $(this)
        , $parent
        , isActive

46
47
      if ($this.is('.disabled, :disabled')) return

48
      $parent = getParent($this)
49

Jacob Thornton's avatar
Jacob Thornton committed
50
51
52
      isActive = $parent.hasClass('open')

      clearMenus()
53

54
55
      if (!isActive) {
        $parent.toggleClass('open')
Jacob Thornton's avatar
Jacob Thornton committed
56
57
      }

fat's avatar
fat committed
58
59
      $this.focus()

Jacob Thornton's avatar
Jacob Thornton committed
60
61
62
      return false
    }

63
  , keydown: function (e) {
64
      var $this
65
66
67
        , $items
        , $active
        , $parent
68
        , isActive
69
70
        , index

71
      if (!/(38|40|27)/.test(e.keyCode)) return
72

73
      $this = $(this)
74
75
76
77
78
79
80

      e.preventDefault()
      e.stopPropagation()

      if ($this.is('.disabled, :disabled')) return

      $parent = getParent($this)
Jacob Thornton's avatar
Jacob Thornton committed
81
82
83

      isActive = $parent.hasClass('open')

84
      if (!isActive || (isActive && e.keyCode == 27)) return $this.click()
85

86
      $items = $('[role=menu] li:not(.divider):visible a', $parent)
Jacob Thornton's avatar
Jacob Thornton committed
87

88
89
      if (!$items.length) return

90
      index = $items.index($items.filter(':focus'))
91
92

      if (e.keyCode == 38 && index > 0) index--                                        // up
93
      if (e.keyCode == 40 && index < $items.length - 1) index++                        // down
94
95
96
97
      if (!~index) index = 0

      $items
        .eq(index)
98
        .focus()
Jacob Thornton's avatar
Jacob Thornton committed
99
100
101
102
    }

  }

Mark Otto's avatar
Mark Otto committed
103
104
  function clearMenus() {
    $(toggle).each(function () {
fat's avatar
fat committed
105
      getParent($(this)).removeClass('open')
106
    })
107
108
109
110
111
112
113
114
  }

  function getParent($this) {
    var selector = $this.attr('data-target')
      , $parent

    if (!selector) {
      selector = $this.attr('href')
115
      selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
116
117
    }

118
119
120
    $parent = selector && $(selector)

    if (!$parent || !$parent.length) $parent = $this.parent()
121
122

    return $parent
Jacob Thornton's avatar
Jacob Thornton committed
123
124
125
126
127
128
  }


  /* DROPDOWN PLUGIN DEFINITION
   * ========================== */

129
130
  var old = $.fn.dropdown

131
  $.fn.dropdown = function (option) {
Jacob Thornton's avatar
Jacob Thornton committed
132
133
134
135
136
137
138
139
140
141
142
    return this.each(function () {
      var $this = $(this)
        , data = $this.data('dropdown')
      if (!data) $this.data('dropdown', (data = new Dropdown(this)))
      if (typeof option == 'string') data[option].call($this)
    })
  }

  $.fn.dropdown.Constructor = Dropdown


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

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


Jacob Thornton's avatar
Jacob Thornton committed
152
153
154
  /* APPLY TO STANDARD DROPDOWN ELEMENTS
   * =================================== */

155
156
157
  $(document)
    .on('click.dropdown.data-api touchstart.dropdown.data-api', clearMenus)
    .on('click.dropdown touchstart.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
158
    .on('touchstart.dropdown.data-api', '.dropdown-menu', function (e) { e.stopPropagation() })
159
160
    .on('click.dropdown.data-api touchstart.dropdown.data-api'  , toggle, Dropdown.prototype.toggle)
    .on('keydown.dropdown.data-api touchstart.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
Jacob Thornton's avatar
Jacob Thornton committed
161

162
}(window.jQuery);