mirror of
https://github.com/twbs/bootstrap.git
synced 2025-03-15 15:29:22 +01:00
The current config can cause the "body" to become stale. Specifically, if the entire body element is swapped out for a new body element, then the backdrop will continue to append itself to the original body element, since it's stored in memory as a reference on this object. This also no longer allows an explicit null to be passed to Backdrop's rootElement This still accomplishes the laziness of "not finding the rootElement until the Backdrop is created" to avoid problems of the JavaScript being included inside <head> (so, before body is available).
137 lines
3.2 KiB
JavaScript
137 lines
3.2 KiB
JavaScript
/**
|
|
* --------------------------------------------------------------------------
|
|
* Bootstrap (v5.0.1): util/backdrop.js
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* --------------------------------------------------------------------------
|
|
*/
|
|
|
|
import EventHandler from '../dom/event-handler'
|
|
import { emulateTransitionEnd, execute, getElement, getTransitionDurationFromElement, reflow, typeCheckConfig } from './index'
|
|
|
|
const Default = {
|
|
isVisible: true, // if false, we use the backdrop helper without adding any element to the dom
|
|
isAnimated: false,
|
|
rootElement: 'body', // give the choice to place backdrop under different elements
|
|
clickCallback: null
|
|
}
|
|
|
|
const DefaultType = {
|
|
isVisible: 'boolean',
|
|
isAnimated: 'boolean',
|
|
rootElement: '(element|string)',
|
|
clickCallback: '(function|null)'
|
|
}
|
|
const NAME = 'backdrop'
|
|
const CLASS_NAME_BACKDROP = 'modal-backdrop'
|
|
const CLASS_NAME_FADE = 'fade'
|
|
const CLASS_NAME_SHOW = 'show'
|
|
|
|
const EVENT_MOUSEDOWN = `mousedown.bs.${NAME}`
|
|
|
|
class Backdrop {
|
|
constructor(config) {
|
|
this._config = this._getConfig(config)
|
|
this._isAppended = false
|
|
this._element = null
|
|
}
|
|
|
|
show(callback) {
|
|
if (!this._config.isVisible) {
|
|
execute(callback)
|
|
return
|
|
}
|
|
|
|
this._append()
|
|
|
|
if (this._config.isAnimated) {
|
|
reflow(this._getElement())
|
|
}
|
|
|
|
this._getElement().classList.add(CLASS_NAME_SHOW)
|
|
|
|
this._emulateAnimation(() => {
|
|
execute(callback)
|
|
})
|
|
}
|
|
|
|
hide(callback) {
|
|
if (!this._config.isVisible) {
|
|
execute(callback)
|
|
return
|
|
}
|
|
|
|
this._getElement().classList.remove(CLASS_NAME_SHOW)
|
|
|
|
this._emulateAnimation(() => {
|
|
this.dispose()
|
|
execute(callback)
|
|
})
|
|
}
|
|
|
|
// Private
|
|
|
|
_getElement() {
|
|
if (!this._element) {
|
|
const backdrop = document.createElement('div')
|
|
backdrop.className = CLASS_NAME_BACKDROP
|
|
if (this._config.isAnimated) {
|
|
backdrop.classList.add(CLASS_NAME_FADE)
|
|
}
|
|
|
|
this._element = backdrop
|
|
}
|
|
|
|
return this._element
|
|
}
|
|
|
|
_getConfig(config) {
|
|
config = {
|
|
...Default,
|
|
...(typeof config === 'object' ? config : {})
|
|
}
|
|
|
|
// use getElement() with the default "body" to get a fresh Element on each instantiation
|
|
config.rootElement = getElement(config.rootElement)
|
|
typeCheckConfig(NAME, config, DefaultType)
|
|
return config
|
|
}
|
|
|
|
_append() {
|
|
if (this._isAppended) {
|
|
return
|
|
}
|
|
|
|
this._config.rootElement.appendChild(this._getElement())
|
|
|
|
EventHandler.on(this._getElement(), EVENT_MOUSEDOWN, () => {
|
|
execute(this._config.clickCallback)
|
|
})
|
|
|
|
this._isAppended = true
|
|
}
|
|
|
|
dispose() {
|
|
if (!this._isAppended) {
|
|
return
|
|
}
|
|
|
|
EventHandler.off(this._element, EVENT_MOUSEDOWN)
|
|
|
|
this._element.remove()
|
|
this._isAppended = false
|
|
}
|
|
|
|
_emulateAnimation(callback) {
|
|
if (!this._config.isAnimated) {
|
|
execute(callback)
|
|
return
|
|
}
|
|
|
|
const backdropTransitionDuration = getTransitionDurationFromElement(this._getElement())
|
|
EventHandler.one(this._getElement(), 'transitionend', () => execute(callback))
|
|
emulateTransitionEnd(this._getElement(), backdropTransitionDuration)
|
|
}
|
|
}
|
|
|
|
export default Backdrop
|