mirror of
https://github.com/twbs/bootstrap.git
synced 2025-03-14 14:29:30 +01:00
150 lines
3.1 KiB
JavaScript
150 lines
3.1 KiB
JavaScript
/**
|
|
* --------------------------------------------------------------------------
|
|
* Bootstrap (v5.2.0-beta1): util/backdrop.js
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
|
|
* --------------------------------------------------------------------------
|
|
*/
|
|
|
|
import EventHandler from '../dom/event-handler'
|
|
import { execute, executeAfterTransition, getElement, reflow } from './index'
|
|
import Config from './config'
|
|
|
|
/**
|
|
* Constants
|
|
*/
|
|
|
|
const NAME = 'backdrop'
|
|
const CLASS_NAME_FADE = 'fade'
|
|
const CLASS_NAME_SHOW = 'show'
|
|
const EVENT_MOUSEDOWN = `mousedown.bs.${NAME}`
|
|
|
|
const Default = {
|
|
className: 'modal-backdrop',
|
|
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 = {
|
|
className: 'string',
|
|
isVisible: 'boolean',
|
|
isAnimated: 'boolean',
|
|
rootElement: '(element|string)',
|
|
clickCallback: '(function|null)'
|
|
}
|
|
|
|
/**
|
|
* Class definition
|
|
*/
|
|
|
|
class Backdrop extends Config {
|
|
constructor(config) {
|
|
super()
|
|
this._config = this._getConfig(config)
|
|
this._isAppended = false
|
|
this._element = null
|
|
}
|
|
|
|
// Getters
|
|
static get Default() {
|
|
return Default
|
|
}
|
|
|
|
static get DefaultType() {
|
|
return DefaultType
|
|
}
|
|
|
|
static get NAME() {
|
|
return NAME
|
|
}
|
|
|
|
// Public
|
|
show(callback) {
|
|
if (!this._config.isVisible) {
|
|
execute(callback)
|
|
return
|
|
}
|
|
|
|
this._append()
|
|
|
|
const element = this._getElement()
|
|
if (this._config.isAnimated) {
|
|
reflow(element)
|
|
}
|
|
|
|
element.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)
|
|
})
|
|
}
|
|
|
|
dispose() {
|
|
if (!this._isAppended) {
|
|
return
|
|
}
|
|
|
|
EventHandler.off(this._element, EVENT_MOUSEDOWN)
|
|
|
|
this._element.remove()
|
|
this._isAppended = false
|
|
}
|
|
|
|
// Private
|
|
_getElement() {
|
|
if (!this._element) {
|
|
const backdrop = document.createElement('div')
|
|
backdrop.className = this._config.className
|
|
if (this._config.isAnimated) {
|
|
backdrop.classList.add(CLASS_NAME_FADE)
|
|
}
|
|
|
|
this._element = backdrop
|
|
}
|
|
|
|
return this._element
|
|
}
|
|
|
|
_configAfterMerge(config) {
|
|
// use getElement() with the default "body" to get a fresh Element on each instantiation
|
|
config.rootElement = getElement(config.rootElement)
|
|
return config
|
|
}
|
|
|
|
_append() {
|
|
if (this._isAppended) {
|
|
return
|
|
}
|
|
|
|
const element = this._getElement()
|
|
this._config.rootElement.append(element)
|
|
|
|
EventHandler.on(element, EVENT_MOUSEDOWN, () => {
|
|
execute(this._config.clickCallback)
|
|
})
|
|
|
|
this._isAppended = true
|
|
}
|
|
|
|
_emulateAnimation(callback) {
|
|
executeAfterTransition(callback, this._getElement(), this._config.isAnimated)
|
|
}
|
|
}
|
|
|
|
export default Backdrop
|