bootstrap-modal.js 6.69 KB
Newer Older
Jacob Thornton's avatar
Jacob Thornton committed
1
/* =========================================================
Mark Otto's avatar
Mark Otto committed
2
 * bootstrap-modal.js v3.0.0
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#modals
 * =========================================================
 * 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


 /* MODAL CLASS DEFINITION
  * ====================== */

Jacob Thornton's avatar
Jacob Thornton committed
29
  var Modal = function (element, options) {
30
    this.options = options
Jacob Thornton's avatar
Jacob Thornton committed
31
    this.$element = $(element)
Jacob Thornton's avatar
Jacob Thornton committed
32
      .delegate('[data-dismiss="modal"]', 'click.dismiss.modal', $.proxy(this.hide, this))
33
    this.options.remote && this.$element.find('.modal-body').load(this.options.remote)
Jacob Thornton's avatar
Jacob Thornton committed
34
35
36
37
38
39
40
41
42
43
44
45
  }

  Modal.prototype = {

      constructor: Modal

    , toggle: function () {
        return this[!this.isShown ? 'show' : 'hide']()
      }

    , show: function () {
        var that = this
46
          , e = $.Event('show')
Jacob Thornton's avatar
Jacob Thornton committed
47

48
49
50
        this.$element.trigger(e)

        if (this.isShown || e.isDefaultPrevented()) return
Jacob Thornton's avatar
Jacob Thornton committed
51
52
53

        this.isShown = true

54
55
56
        this.escape()

        this.backdrop(function () {
Jacob Thornton's avatar
Jacob Thornton committed
57
58
          var transition = $.support.transition && that.$element.hasClass('fade')

59
60
61
          if (!that.$element.parent().length) {
            that.$element.appendTo(document.body) //don't move modals dom position
          }
Jacob Thornton's avatar
Jacob Thornton committed
62

fat's avatar
fat committed
63
          that.$element.show()
Jacob Thornton's avatar
Jacob Thornton committed
64
65
66
67
68

          if (transition) {
            that.$element[0].offsetWidth // force reflow
          }

69
70
71
          that.$element
            .addClass('in')
            .attr('aria-hidden', false)
Jacob Thornton's avatar
Jacob Thornton committed
72

73
74
          that.enforceFocus()

Jacob Thornton's avatar
Jacob Thornton committed
75
          transition ?
76
77
            that.$element.one($.support.transition.end, function () { that.$element.focus().trigger('shown') }) :
            that.$element.focus().trigger('shown')
Jacob Thornton's avatar
Jacob Thornton committed
78
79
80
81

        })
      }

82
    , hide: function (e) {
Jacob Thornton's avatar
Jacob Thornton committed
83
84
85
        e && e.preventDefault()

        var that = this
86
87
88
89
90
91
92

        e = $.Event('hide')

        this.$element.trigger(e)

        if (!this.isShown || e.isDefaultPrevented()) return

Jacob Thornton's avatar
Jacob Thornton committed
93
94
        this.isShown = false

95
        this.escape()
96
97

        $(document).off('focusin.modal')
Jacob Thornton's avatar
Jacob Thornton committed
98

99
100
101
        this.$element
          .removeClass('in')
          .attr('aria-hidden', true)
Jacob Thornton's avatar
Jacob Thornton committed
102
103

        $.support.transition && this.$element.hasClass('fade') ?
104
105
          this.hideWithTransition() :
          this.hideModal()
Jacob Thornton's avatar
Jacob Thornton committed
106
107
      }

108
109
110
111
112
113
114
115
    , enforceFocus: function () {
        var that = this
        $(document).on('focusin.modal', function (e) {
          if (that.$element[0] !== e.target && !that.$element.has(e.target).length) {
            that.$element.focus()
          }
        })
      }
Jacob Thornton's avatar
Jacob Thornton committed
116

117
118
119
    , escape: function () {
        var that = this
        if (this.isShown && this.options.keyboard) {
120
          this.$element.on('keyup.dismiss.modal', function ( e ) {
121
122
123
            e.which == 27 && that.hide()
          })
        } else if (!this.isShown) {
124
          this.$element.off('keyup.dismiss.modal')
125
126
        }
      }
Jacob Thornton's avatar
Jacob Thornton committed
127

128
129
130
131
132
133
134
135
136
137
138
139
    , hideWithTransition: function () {
        var that = this
          , timeout = setTimeout(function () {
              that.$element.off($.support.transition.end)
              that.hideModal()
            }, 500)

        this.$element.one($.support.transition.end, function () {
          clearTimeout(timeout)
          that.hideModal()
        })
      }
Jacob Thornton's avatar
Jacob Thornton committed
140

141
142
143
144
145
146
147
    , hideModal: function () {
        var that = this
        this.$element.hide()
        this.backdrop(function () {
          that.removeBackdrop()
          that.$element.trigger('hidden')
        })
148
      }
Jacob Thornton's avatar
Jacob Thornton committed
149

150
    , removeBackdrop: function () {
fat's avatar
shiiiit    
fat committed
151
        this.$backdrop && this.$backdrop.remove()
152
153
        this.$backdrop = null
      }
Jacob Thornton's avatar
Jacob Thornton committed
154

155
156
157
    , backdrop: function (callback) {
        var that = this
          , animate = this.$element.hasClass('fade') ? 'fade' : ''
Jacob Thornton's avatar
Jacob Thornton committed
158

159
160
        if (this.isShown && this.options.backdrop) {
          var doAnimate = $.support.transition && animate
Jacob Thornton's avatar
Jacob Thornton committed
161

162
163
          this.$backdrop = $('<div class="modal-backdrop ' + animate + '" />')
            .appendTo(document.body)
Jacob Thornton's avatar
Jacob Thornton committed
164

Jacob Thornton's avatar
Jacob Thornton committed
165
166
167
168
169
          this.$backdrop.click(
            this.options.backdrop == 'static' ?
              $.proxy(this.$element[0].focus, this.$element[0])
            : $.proxy(this.hide, this)
          )
Jacob Thornton's avatar
Jacob Thornton committed
170

171
          if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
Jacob Thornton's avatar
Jacob Thornton committed
172

173
          this.$backdrop.addClass('in')
Jacob Thornton's avatar
Jacob Thornton committed
174

175
176
          if (!callback) return

177
178
179
          doAnimate ?
            this.$backdrop.one($.support.transition.end, callback) :
            callback()
Jacob Thornton's avatar
Jacob Thornton committed
180

181
182
        } else if (!this.isShown && this.$backdrop) {
          this.$backdrop.removeClass('in')
Jacob Thornton's avatar
Jacob Thornton committed
183

184
          $.support.transition && this.$element.hasClass('fade')?
185
186
            this.$backdrop.one($.support.transition.end, callback) :
            callback()
Jacob Thornton's avatar
Jacob Thornton committed
187

188
189
190
191
        } else if (callback) {
          callback()
        }
      }
Jacob Thornton's avatar
Jacob Thornton committed
192
193
194
195
196
197
  }


 /* MODAL PLUGIN DEFINITION
  * ======================= */

198
199
  var old = $.fn.modal

200
  $.fn.modal = function (option) {
Jacob Thornton's avatar
Jacob Thornton committed
201
202
203
    return this.each(function () {
      var $this = $(this)
        , data = $this.data('modal')
204
        , options = $.extend({}, $.fn.modal.defaults, $this.data(), typeof option == 'object' && option)
Jacob Thornton's avatar
Jacob Thornton committed
205
206
      if (!data) $this.data('modal', (data = new Modal(this, options)))
      if (typeof option == 'string') data[option]()
207
      else if (options.show) data.show()
Jacob Thornton's avatar
Jacob Thornton committed
208
209
210
211
212
213
    })
  }

  $.fn.modal.defaults = {
      backdrop: true
    , keyboard: true
214
    , show: true
Jacob Thornton's avatar
Jacob Thornton committed
215
216
217
218
219
  }

  $.fn.modal.Constructor = Modal


220
221
222
223
224
225
226
227
228
 /* MODAL NO CONFLICT
  * ================= */

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


Jacob Thornton's avatar
Jacob Thornton committed
229
230
231
 /* MODAL DATA-API
  * ============== */

232
233
234
235
236
  $(document).on('click.modal.data-api', '[data-toggle="modal"]', function (e) {
    var $this = $(this)
      , href = $this.attr('href')
      , $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) //strip for ie7
      , option = $target.data('modal') ? 'toggle' : $.extend({ remote:!/#/.test(href) && href }, $target.data(), $this.data())
Jacob Thornton's avatar
Jacob Thornton committed
237

238
    e.preventDefault()
239

240
241
242
243
244
    $target
      .modal(option)
      .one('hide', function () {
        $this.focus()
      })
245
246
    })

247
    var $body = $(document.body)
248
249
      .on('shown', '.modal', function () { $body.addClass('modal-open') })
      .on('hidden', '.modal', function () { $body.removeClass('modal-open') })
Jacob Thornton's avatar
Jacob Thornton committed
250

251
}(window.jQuery);