modal.js 16.6 KB
Newer Older
1
import $ from 'jquery'
2
3
4
5
import Util from './util'

/**
 * --------------------------------------------------------------------------
Mark Otto's avatar
Mark Otto committed
6
 * Bootstrap (v4.0.0): modal.js
7
8
9
10
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
 * --------------------------------------------------------------------------
 */

11
const Modal = (($) => {
12
13
14
15
16
17
  /**
   * ------------------------------------------------------------------------
   * Constants
   * ------------------------------------------------------------------------
   */

18
19
20
21
22
23
24
  const NAME               = 'modal'
  const VERSION            = '4.0.0'
  const DATA_KEY           = 'bs.modal'
  const EVENT_KEY          = `.${DATA_KEY}`
  const DATA_API_KEY       = '.data-api'
  const JQUERY_NO_CONFLICT = $.fn[NAME]
  const ESCAPE_KEYCODE     = 27 // KeyboardEvent.which value for Escape (Esc) key
25
26
27
28

  const Default = {
    backdrop : true,
    keyboard : true,
29
    focus    : true,
30
31
32
    show     : true
  }

fat's avatar
fat committed
33
34
35
36
37
38
39
  const DefaultType = {
    backdrop : '(boolean|string)',
    keyboard : 'boolean',
    focus    : 'boolean',
    show     : 'boolean'
  }

40
  const Event = {
XhmikosR's avatar
XhmikosR committed
41
42
43
44
    HIDE              : `hide${EVENT_KEY}`,
    HIDDEN            : `hidden${EVENT_KEY}`,
    SHOW              : `show${EVENT_KEY}`,
    SHOWN             : `shown${EVENT_KEY}`,
fat's avatar
fat committed
45
46
47
48
49
50
51
    FOCUSIN           : `focusin${EVENT_KEY}`,
    RESIZE            : `resize${EVENT_KEY}`,
    CLICK_DISMISS     : `click.dismiss${EVENT_KEY}`,
    KEYDOWN_DISMISS   : `keydown.dismiss${EVENT_KEY}`,
    MOUSEUP_DISMISS   : `mouseup.dismiss${EVENT_KEY}`,
    MOUSEDOWN_DISMISS : `mousedown.dismiss${EVENT_KEY}`,
    CLICK_DATA_API    : `click${EVENT_KEY}${DATA_API_KEY}`
52
53
54
  }

  const ClassName = {
55
56
57
58
    SCROLLBAR_MEASURER : 'modal-scrollbar-measure',
    BACKDROP           : 'modal-backdrop',
    OPEN               : 'modal-open',
    FADE               : 'fade',
Starsam80's avatar
Starsam80 committed
59
    SHOW               : 'show'
60
61
62
63
64
65
  }

  const Selector = {
    DIALOG             : '.modal-dialog',
    DATA_TOGGLE        : '[data-toggle="modal"]',
    DATA_DISMISS       : '[data-dismiss="modal"]',
66
    FIXED_CONTENT      : '.fixed-top, .fixed-bottom, .is-fixed, .sticky-top',
67
    STICKY_CONTENT     : '.sticky-top',
68
    NAVBAR_TOGGLER     : '.navbar-toggler'
69
70
71
72
73
74
75
76
77
78
  }

  /**
   * ------------------------------------------------------------------------
   * Class Definition
   * ------------------------------------------------------------------------
   */

  class Modal {
    constructor(element, config) {
fat's avatar
fat committed
79
      this._config              = this._getConfig(config)
80
81
82
83
84
85
86
87
88
      this._element             = element
      this._dialog              = $(element).find(Selector.DIALOG)[0]
      this._backdrop            = null
      this._isShown             = false
      this._isBodyOverflowing   = false
      this._ignoreBackdropClick = false
      this._scrollbarWidth      = 0
    }

XhmikosR's avatar
XhmikosR committed
89
    // Getters
90
91
92
93
94
95
96
97
98

    static get VERSION() {
      return VERSION
    }

    static get Default() {
      return Default
    }

XhmikosR's avatar
XhmikosR committed
99
    // Public
100
101
102
103
104
105

    toggle(relatedTarget) {
      return this._isShown ? this.hide() : this.show(relatedTarget)
    }

    show(relatedTarget) {
lucascono's avatar
lucascono committed
106
      if (this._isTransitioning || this._isShown) {
107
        return
108
109
      }

110
      if (Util.supportsTransitionEnd() && $(this._element).hasClass(ClassName.FADE)) {
111
112
        this._isTransitioning = true
      }
113

114
      const showEvent = $.Event(Event.SHOW, {
Jacob Thornton's avatar
Jacob Thornton committed
115
        relatedTarget
116
117
118
119
120
121
122
123
124
125
126
127
128
      })

      $(this._element).trigger(showEvent)

      if (this._isShown || showEvent.isDefaultPrevented()) {
        return
      }

      this._isShown = true

      this._checkScrollbar()
      this._setScrollbar()

David Bailey's avatar
David Bailey committed
129
130
      this._adjustDialog()

131
132
133
134
135
136
      $(document.body).addClass(ClassName.OPEN)

      this._setEscapeEvent()
      this._setResizeEvent()

      $(this._element).on(
fat's avatar
fat committed
137
        Event.CLICK_DISMISS,
138
        Selector.DATA_DISMISS,
139
        (event) => this.hide(event)
140
141
      )

fat's avatar
fat committed
142
143
      $(this._dialog).on(Event.MOUSEDOWN_DISMISS, () => {
        $(this._element).one(Event.MOUSEUP_DISMISS, (event) => {
144
          if ($(event.target).is(this._element)) {
145
            this._ignoreBackdropClick = true
146
147
148
149
          }
        })
      })

150
      this._showBackdrop(() => this._showElement(relatedTarget))
151
152
153
154
155
156
157
    }

    hide(event) {
      if (event) {
        event.preventDefault()
      }

158
159
      if (this._isTransitioning || !this._isShown) {
        return
160
      }
161

162
      const hideEvent = $.Event(Event.HIDE)
163

164
165
166
167
168
169
170
171
      $(this._element).trigger(hideEvent)

      if (!this._isShown || hideEvent.isDefaultPrevented()) {
        return
      }

      this._isShown = false

172
173
174
175
176
177
      const transition = Util.supportsTransitionEnd() && $(this._element).hasClass(ClassName.FADE)

      if (transition) {
        this._isTransitioning = true
      }

178
179
180
181
182
      this._setEscapeEvent()
      this._setResizeEvent()

      $(document).off(Event.FOCUSIN)

Starsam80's avatar
Starsam80 committed
183
      $(this._element).removeClass(ClassName.SHOW)
184

fat's avatar
fat committed
185
186
      $(this._element).off(Event.CLICK_DISMISS)
      $(this._dialog).off(Event.MOUSEDOWN_DISMISS)
187

188

189
      if (transition) {
190
191
        const transitionDuration  = Util.getTransitionDurationFromElement(this._element)

192
        $(this._element)
193
          .one(Util.TRANSITION_END, (event) => this._hideModal(event))
194
          .emulateTransitionEnd(transitionDuration)
195
196
197
198
199
      } else {
        this._hideModal()
      }
    }

fat's avatar
fat committed
200
201
202
    dispose() {
      $.removeData(this._element, DATA_KEY)

203
      $(window, document, this._element, this._backdrop).off(EVENT_KEY)
fat's avatar
fat committed
204
205
206
207
208
209
210
211
212
213
214

      this._config              = null
      this._element             = null
      this._dialog              = null
      this._backdrop            = null
      this._isShown             = null
      this._isBodyOverflowing   = null
      this._ignoreBackdropClick = null
      this._scrollbarWidth      = null
    }

215
216
217
    handleUpdate() {
      this._adjustDialog()
    }
218

XhmikosR's avatar
XhmikosR committed
219
    // Private
220

fat's avatar
fat committed
221
    _getConfig(config) {
222
223
224
225
      config = {
        ...Default,
        ...config
      }
fat's avatar
fat committed
226
227
228
229
      Util.typeCheckConfig(NAME, config, DefaultType)
      return config
    }

230
    _showElement(relatedTarget) {
231
      const transition = Util.supportsTransitionEnd() &&
232
233
234
        $(this._element).hasClass(ClassName.FADE)

      if (!this._element.parentNode ||
235
         this._element.parentNode.nodeType !== Node.ELEMENT_NODE) {
XhmikosR's avatar
XhmikosR committed
236
        // Don't move modal's DOM position
237
238
239
240
        document.body.appendChild(this._element)
      }

      this._element.style.display = 'block'
241
      this._element.removeAttribute('aria-hidden')
242
243
244
245
246
247
      this._element.scrollTop = 0

      if (transition) {
        Util.reflow(this._element)
      }

Starsam80's avatar
Starsam80 committed
248
      $(this._element).addClass(ClassName.SHOW)
249

Jacob Thornton's avatar
Jacob Thornton committed
250
251
252
      if (this._config.focus) {
        this._enforceFocus()
      }
253

254
      const shownEvent = $.Event(Event.SHOWN, {
Jacob Thornton's avatar
Jacob Thornton committed
255
        relatedTarget
256
257
      })

258
      const transitionComplete = () => {
Jacob Thornton's avatar
Jacob Thornton committed
259
260
261
        if (this._config.focus) {
          this._element.focus()
        }
262
        this._isTransitioning = false
263
264
265
266
        $(this._element).trigger(shownEvent)
      }

      if (transition) {
267
268
        const transitionDuration  = Util.getTransitionDurationFromElement(this._element)

269
270
        $(this._dialog)
          .one(Util.TRANSITION_END, transitionComplete)
271
          .emulateTransitionEnd(transitionDuration)
272
273
274
275
276
277
278
      } else {
        transitionComplete()
      }
    }

    _enforceFocus() {
      $(document)
XhmikosR's avatar
XhmikosR committed
279
        .off(Event.FOCUSIN) // Guard against infinite focus loop
280
        .on(Event.FOCUSIN, (event) => {
281
282
          if (document !== event.target &&
              this._element !== event.target &&
XhmikosR's avatar
XhmikosR committed
283
              $(this._element).has(event.target).length === 0) {
284
285
286
287
288
289
290
            this._element.focus()
          }
        })
    }

    _setEscapeEvent() {
      if (this._isShown && this._config.keyboard) {
fat's avatar
fat committed
291
        $(this._element).on(Event.KEYDOWN_DISMISS, (event) => {
292
          if (event.which === ESCAPE_KEYCODE) {
293
            event.preventDefault()
294
295
296
297
            this.hide()
          }
        })
      } else if (!this._isShown) {
fat's avatar
fat committed
298
        $(this._element).off(Event.KEYDOWN_DISMISS)
299
300
301
302
303
      }
    }

    _setResizeEvent() {
      if (this._isShown) {
304
        $(window).on(Event.RESIZE, (event) => this.handleUpdate(event))
305
306
307
308
309
310
311
      } else {
        $(window).off(Event.RESIZE)
      }
    }

    _hideModal() {
      this._element.style.display = 'none'
312
      this._element.setAttribute('aria-hidden', true)
313
      this._isTransitioning = false
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
      this._showBackdrop(() => {
        $(document.body).removeClass(ClassName.OPEN)
        this._resetAdjustments()
        this._resetScrollbar()
        $(this._element).trigger(Event.HIDDEN)
      })
    }

    _removeBackdrop() {
      if (this._backdrop) {
        $(this._backdrop).remove()
        this._backdrop = null
      }
    }

    _showBackdrop(callback) {
XhmikosR's avatar
XhmikosR committed
330
331
      const animate = $(this._element).hasClass(ClassName.FADE)
        ? ClassName.FADE : ''
332
333

      if (this._isShown && this._config.backdrop) {
334
        const doAnimate = Util.supportsTransitionEnd() && animate
335
336
337
338
339
340
341
342

        this._backdrop = document.createElement('div')
        this._backdrop.className = ClassName.BACKDROP

        if (animate) {
          $(this._backdrop).addClass(animate)
        }

Jacob Thornton's avatar
Jacob Thornton committed
343
        $(this._backdrop).appendTo(document.body)
344

fat's avatar
fat committed
345
        $(this._element).on(Event.CLICK_DISMISS, (event) => {
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
          if (this._ignoreBackdropClick) {
            this._ignoreBackdropClick = false
            return
          }
          if (event.target !== event.currentTarget) {
            return
          }
          if (this._config.backdrop === 'static') {
            this._element.focus()
          } else {
            this.hide()
          }
        })

        if (doAnimate) {
          Util.reflow(this._backdrop)
        }

Starsam80's avatar
Starsam80 committed
364
        $(this._backdrop).addClass(ClassName.SHOW)
365
366
367
368
369
370
371
372
373
374

        if (!callback) {
          return
        }

        if (!doAnimate) {
          callback()
          return
        }

375
376
        const backdropTransitionDuration = Util.getTransitionDurationFromElement(this._backdrop)

377
378
        $(this._backdrop)
          .one(Util.TRANSITION_END, callback)
379
          .emulateTransitionEnd(backdropTransitionDuration)
380
      } else if (!this._isShown && this._backdrop) {
Starsam80's avatar
Starsam80 committed
381
        $(this._backdrop).removeClass(ClassName.SHOW)
382

383
        const callbackRemove = () => {
384
385
386
387
388
389
390
          this._removeBackdrop()
          if (callback) {
            callback()
          }
        }

        if (Util.supportsTransitionEnd() &&
391
           $(this._element).hasClass(ClassName.FADE)) {
392
393
          const backdropTransitionDuration = Util.getTransitionDurationFromElement(this._backdrop)

394
395
          $(this._backdrop)
            .one(Util.TRANSITION_END, callbackRemove)
396
            .emulateTransitionEnd(backdropTransitionDuration)
397
398
399
400
401
402
403
404
405
406
407
408
409
410
        } else {
          callbackRemove()
        }
      } else if (callback) {
        callback()
      }
    }

    // ----------------------------------------------------------------------
    // the following methods are used to handle overflowing modals
    // todo (fat): these should probably be refactored out of modal.js
    // ----------------------------------------------------------------------

    _adjustDialog() {
411
      const isModalOverflowing =
412
413
414
        this._element.scrollHeight > document.documentElement.clientHeight

      if (!this._isBodyOverflowing && isModalOverflowing) {
Jacob Thornton's avatar
Jacob Thornton committed
415
        this._element.style.paddingLeft = `${this._scrollbarWidth}px`
416
417
418
      }

      if (this._isBodyOverflowing && !isModalOverflowing) {
419
        this._element.style.paddingRight = `${this._scrollbarWidth}px`
420
421
422
423
424
425
426
427
428
      }
    }

    _resetAdjustments() {
      this._element.style.paddingLeft = ''
      this._element.style.paddingRight = ''
    }

    _checkScrollbar() {
429
430
      const rect = document.body.getBoundingClientRect()
      this._isBodyOverflowing = rect.left + rect.right < window.innerWidth
431
432
433
434
      this._scrollbarWidth = this._getScrollbarWidth()
    }

    _setScrollbar() {
435
436
437
438
439
440
441
442
443
444
      if (this._isBodyOverflowing) {
        // Note: DOMNode.style.paddingRight returns the actual value or '' if not set
        //   while $(DOMNode).css('padding-right') returns the calculated value or 0 if not set

        // Adjust fixed content padding
        $(Selector.FIXED_CONTENT).each((index, element) => {
          const actualPadding = $(element)[0].style.paddingRight
          const calculatedPadding = $(element).css('padding-right')
          $(element).data('padding-right', actualPadding).css('padding-right', `${parseFloat(calculatedPadding) + this._scrollbarWidth}px`)
        })
445

446
447
448
449
450
451
452
        // Adjust sticky content margin
        $(Selector.STICKY_CONTENT).each((index, element) => {
          const actualMargin = $(element)[0].style.marginRight
          const calculatedMargin = $(element).css('margin-right')
          $(element).data('margin-right', actualMargin).css('margin-right', `${parseFloat(calculatedMargin) - this._scrollbarWidth}px`)
        })

453
454
455
456
457
458
        // Adjust navbar-toggler margin
        $(Selector.NAVBAR_TOGGLER).each((index, element) => {
          const actualMargin = $(element)[0].style.marginRight
          const calculatedMargin = $(element).css('margin-right')
          $(element).data('margin-right', actualMargin).css('margin-right', `${parseFloat(calculatedMargin) + this._scrollbarWidth}px`)
        })
459

460
461
        // Adjust body padding
        const actualPadding = document.body.style.paddingRight
462
463
        const calculatedPadding = $(document.body).css('padding-right')
        $(document.body).data('padding-right', actualPadding).css('padding-right', `${parseFloat(calculatedPadding) + this._scrollbarWidth}px`)
464
465
466
467
      }
    }

    _resetScrollbar() {
468
469
470
471
472
473
474
475
      // Restore fixed content padding
      $(Selector.FIXED_CONTENT).each((index, element) => {
        const padding = $(element).data('padding-right')
        if (typeof padding !== 'undefined') {
          $(element).css('padding-right', padding).removeData('padding-right')
        }
      })

476
477
      // Restore sticky content and navbar-toggler margin
      $(`${Selector.STICKY_CONTENT}, ${Selector.NAVBAR_TOGGLER}`).each((index, element) => {
478
479
480
481
482
483
484
        const margin = $(element).data('margin-right')
        if (typeof margin !== 'undefined') {
          $(element).css('margin-right', margin).removeData('margin-right')
        }
      })

      // Restore body padding
485
      const padding = $(document.body).data('padding-right')
486
      if (typeof padding !== 'undefined') {
487
        $(document.body).css('padding-right', padding).removeData('padding-right')
488
      }
489
490
491
    }

    _getScrollbarWidth() { // thx d.walsh
492
      const scrollDiv = document.createElement('div')
493
      scrollDiv.className = ClassName.SCROLLBAR_MEASURER
494
      document.body.appendChild(scrollDiv)
495
      const scrollbarWidth = scrollDiv.getBoundingClientRect().width - scrollDiv.clientWidth
496
497
498
499
      document.body.removeChild(scrollDiv)
      return scrollbarWidth
    }

XhmikosR's avatar
XhmikosR committed
500
    // Static
501
502
503

    static _jQueryInterface(config, relatedTarget) {
      return this.each(function () {
XhmikosR's avatar
XhmikosR committed
504
        let data = $(this).data(DATA_KEY)
505
506
507
508
509
        const _config = {
          ...Modal.Default,
          ...$(this).data(),
          ...typeof config === 'object' && config
        }
510
511
512
513
514
515
516

        if (!data) {
          data = new Modal(this, _config)
          $(this).data(DATA_KEY, data)
        }

        if (typeof config === 'string') {
XhmikosR's avatar
XhmikosR committed
517
          if (typeof data[config] === 'undefined') {
XhmikosR's avatar
XhmikosR committed
518
            throw new TypeError(`No method named "${config}"`)
519
          }
520
521
522
523
524
525
526
527
528
529
530
531
532
533
          data[config](relatedTarget)
        } else if (_config.show) {
          data.show(relatedTarget)
        }
      })
    }
  }

  /**
   * ------------------------------------------------------------------------
   * Data Api implementation
   * ------------------------------------------------------------------------
   */

fat's avatar
fat committed
534
  $(document).on(Event.CLICK_DATA_API, Selector.DATA_TOGGLE, function (event) {
535
    let target
536
    const selector = Util.getSelectorFromElement(this)
537
538
539
540
541

    if (selector) {
      target = $(selector)[0]
    }

XhmikosR's avatar
XhmikosR committed
542
543
    const config = $(target).data(DATA_KEY)
      ? 'toggle' : {
544
545
546
        ...$(target).data(),
        ...$(this).data()
      }
547

548
    if (this.tagName === 'A' || this.tagName === 'AREA') {
549
550
551
      event.preventDefault()
    }

552
    const $target = $(target).one(Event.SHOW, (showEvent) => {
553
      if (showEvent.isDefaultPrevented()) {
XhmikosR's avatar
XhmikosR committed
554
        // Only register focus restorer if modal will actually get shown
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
        return
      }

      $target.one(Event.HIDDEN, () => {
        if ($(this).is(':visible')) {
          this.focus()
        }
      })
    })

    Modal._jQueryInterface.call($(target), config, this)
  })

  /**
   * ------------------------------------------------------------------------
   * jQuery
   * ------------------------------------------------------------------------
   */

XhmikosR's avatar
XhmikosR committed
574
  $.fn[NAME] = Modal._jQueryInterface
575
  $.fn[NAME].Constructor = Modal
XhmikosR's avatar
XhmikosR committed
576
  $.fn[NAME].noConflict = function () {
577
578
579
580
581
    $.fn[NAME] = JQUERY_NO_CONFLICT
    return Modal._jQueryInterface
  }

  return Modal
582
})($)
583
584

export default Modal