0
0
mirror of https://github.com/twbs/bootstrap.git synced 2025-02-14 11:54:23 +01:00
Bootstrap/js/src/tab/tab.js

263 lines
6.7 KiB
JavaScript
Raw Normal View History

2015-05-11 12:29:06 -07:00
/**
* --------------------------------------------------------------------------
2019-02-13 18:01:40 +02:00
* Bootstrap (v4.3.1): tab.js
2015-05-11 12:29:06 -07:00
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
* --------------------------------------------------------------------------
*/
import {
jQuery as $,
TRANSITION_END,
emulateTransitionEnd,
getElementFromSelector,
getTransitionDurationFromElement,
makeArray,
reflow
2019-07-22 15:24:17 +02:00
} from '../util/index'
import Data from '../dom/data'
import EventHandler from '../dom/event-handler'
import SelectorEngine from '../dom/selector-engine'
2018-09-26 10:39:01 +02:00
/**
* ------------------------------------------------------------------------
* Constants
* ------------------------------------------------------------------------
*/
2019-02-26 13:20:34 +02:00
const NAME = 'tab'
const VERSION = '4.3.1'
const DATA_KEY = 'bs.tab'
const EVENT_KEY = `.${DATA_KEY}`
const DATA_API_KEY = '.data-api'
2018-09-26 10:39:01 +02:00
const Event = {
2019-02-26 13:20:34 +02:00
HIDE: `hide${EVENT_KEY}`,
HIDDEN: `hidden${EVENT_KEY}`,
SHOW: `show${EVENT_KEY}`,
SHOWN: `shown${EVENT_KEY}`,
CLICK_DATA_API: `click${EVENT_KEY}${DATA_API_KEY}`
2018-09-26 10:39:01 +02:00
}
const ClassName = {
2019-02-26 13:20:34 +02:00
DROPDOWN_MENU: 'dropdown-menu',
ACTIVE: 'active',
DISABLED: 'disabled',
FADE: 'fade',
SHOW: 'show'
2018-09-26 10:39:01 +02:00
}
const Selector = {
2019-02-26 13:20:34 +02:00
DROPDOWN: '.dropdown',
NAV_LIST_GROUP: '.nav, .list-group',
ACTIVE: '.active',
ACTIVE_UL: ':scope > li > .active',
DATA_TOGGLE: '[data-toggle="tab"], [data-toggle="pill"], [data-toggle="list"]',
DROPDOWN_TOGGLE: '.dropdown-toggle',
DROPDOWN_ACTIVE_CHILD: ':scope > .dropdown-menu .active'
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
class Tab {
constructor(element) {
this._element = element
2017-09-26 10:04:31 +02:00
Data.setData(this._element, DATA_KEY, this)
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
// Getters
static get VERSION() {
return VERSION
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
// Public
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
show() {
if (this._element.parentNode &&
2017-09-26 10:04:31 +02:00
this._element.parentNode.nodeType === Node.ELEMENT_NODE &&
this._element.classList.contains(ClassName.ACTIVE) ||
this._element.classList.contains(ClassName.DISABLED)) {
2018-09-26 10:39:01 +02:00
return
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
let previous
const target = getElementFromSelector(this._element)
2017-09-26 10:04:31 +02:00
const listElement = SelectorEngine.closest(this._element, Selector.NAV_LIST_GROUP)
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (listElement) {
const itemSelector = listElement.nodeName === 'UL' || listElement.nodeName === 'OL' ? Selector.ACTIVE_UL : Selector.ACTIVE
previous = makeArray(SelectorEngine.find(itemSelector, listElement))
2018-09-26 10:39:01 +02:00
previous = previous[previous.length - 1]
2015-05-11 12:29:06 -07:00
}
2017-09-26 10:04:31 +02:00
let hideEvent = null
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (previous) {
2017-09-26 10:04:31 +02:00
hideEvent = EventHandler.trigger(previous, Event.HIDE, {
relatedTarget: this._element
})
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2017-09-26 10:04:31 +02:00
const showEvent = EventHandler.trigger(this._element, Event.SHOW, {
relatedTarget: previous
})
2018-09-26 10:39:01 +02:00
2017-09-26 10:04:31 +02:00
if (showEvent.defaultPrevented ||
hideEvent !== null && hideEvent.defaultPrevented) {
2018-09-26 10:39:01 +02:00
return
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
this._activate(
this._element,
listElement
)
const complete = () => {
2017-09-26 10:04:31 +02:00
EventHandler.trigger(previous, Event.HIDDEN, {
2015-05-11 12:29:06 -07:00
relatedTarget: this._element
})
2017-09-26 10:04:31 +02:00
EventHandler.trigger(this._element, Event.SHOWN, {
2015-05-11 12:29:06 -07:00
relatedTarget: previous
})
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (target) {
this._activate(target, target.parentNode, complete)
} else {
complete()
}
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
dispose() {
2017-09-26 10:04:31 +02:00
Data.removeData(this._element, DATA_KEY)
2018-09-26 10:39:01 +02:00
this._element = null
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
// Private
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
_activate(element, container, callback) {
2019-02-26 13:20:34 +02:00
const activeElements = container && (container.nodeName === 'UL' || container.nodeName === 'OL') ?
SelectorEngine.find(Selector.ACTIVE_UL, container) :
SelectorEngine.children(container, Selector.ACTIVE)
2017-09-26 10:04:31 +02:00
2019-02-26 13:20:34 +02:00
const active = activeElements[0]
2017-09-26 10:04:31 +02:00
const isTransitioning = callback &&
(active && active.classList.contains(ClassName.FADE))
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
const complete = () => this._transitionComplete(
element,
active,
callback
)
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (active && isTransitioning) {
const transitionDuration = getTransitionDurationFromElement(active)
2017-09-26 10:04:31 +02:00
active.classList.remove(ClassName.SHOW)
2015-05-11 12:29:06 -07:00
EventHandler.one(active, TRANSITION_END, complete)
emulateTransitionEnd(active, transitionDuration)
2018-09-26 10:39:01 +02:00
} else {
complete()
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
_transitionComplete(element, active, callback) {
if (active) {
2017-09-26 10:04:31 +02:00
active.classList.remove(ClassName.ACTIVE)
2015-05-13 12:48:34 -07:00
2017-09-26 10:04:31 +02:00
const dropdownChild = SelectorEngine.findOne(Selector.DROPDOWN_ACTIVE_CHILD, active.parentNode)
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (dropdownChild) {
2017-09-26 10:04:31 +02:00
dropdownChild.classList.remove(ClassName.ACTIVE)
}
2018-09-26 10:39:01 +02:00
if (active.getAttribute('role') === 'tab') {
active.setAttribute('aria-selected', false)
2015-05-11 12:29:06 -07:00
}
}
2017-09-26 10:04:31 +02:00
element.classList.add(ClassName.ACTIVE)
2018-09-26 10:39:01 +02:00
if (element.getAttribute('role') === 'tab') {
element.setAttribute('aria-selected', true)
}
2015-08-18 22:03:34 -07:00
reflow(element)
if (element.classList.contains(ClassName.FADE)) {
element.classList.add(ClassName.SHOW)
}
2015-05-11 12:29:06 -07:00
2017-09-26 10:04:31 +02:00
if (element.parentNode && element.parentNode.classList.contains(ClassName.DROPDOWN_MENU)) {
const dropdownElement = SelectorEngine.closest(element, Selector.DROPDOWN)
2018-09-26 10:39:01 +02:00
if (dropdownElement) {
makeArray(SelectorEngine.find(Selector.DROPDOWN_TOGGLE))
2019-02-26 13:20:34 +02:00
.forEach(dropdown => dropdown.classList.add(ClassName.ACTIVE))
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
element.setAttribute('aria-expanded', true)
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (callback) {
callback()
2015-05-11 12:29:06 -07:00
}
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
// Static
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
static _jQueryInterface(config) {
return this.each(function () {
2017-09-26 10:04:31 +02:00
const data = Data.getData(this, DATA_KEY) || new Tab(this)
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
if (typeof config === 'string') {
if (typeof data[config] === 'undefined') {
throw new TypeError(`No method named "${config}"`)
2015-05-11 12:29:06 -07:00
}
2019-02-26 13:20:34 +02:00
2018-09-26 10:39:01 +02:00
data[config]()
}
})
2015-05-11 12:29:06 -07:00
}
static _getInstance(element) {
return Data.getData(element, DATA_KEY)
}
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
/**
* ------------------------------------------------------------------------
* Data Api implementation
* ------------------------------------------------------------------------
*/
2015-05-11 12:29:06 -07:00
2017-09-26 10:04:31 +02:00
EventHandler.on(document, Event.CLICK_DATA_API, Selector.DATA_TOGGLE, function (event) {
event.preventDefault()
const data = Data.getData(this, DATA_KEY) || new Tab(this)
data.show()
})
2015-05-11 12:29:06 -07:00
2018-09-26 10:39:01 +02:00
/**
* ------------------------------------------------------------------------
* jQuery
* ------------------------------------------------------------------------
* add .tab to jQuery only if jQuery is present
2018-09-26 10:39:01 +02:00
*/
2019-07-22 15:24:17 +02:00
/* istanbul ignore if */
2017-09-26 10:04:31 +02:00
if (typeof $ !== 'undefined') {
const JQUERY_NO_CONFLICT = $.fn[NAME]
2019-02-26 13:20:34 +02:00
$.fn[NAME] = Tab._jQueryInterface
$.fn[NAME].Constructor = Tab
$.fn[NAME].noConflict = () => {
2017-09-26 10:04:31 +02:00
$.fn[NAME] = JQUERY_NO_CONFLICT
return Tab._jQueryInterface
}
2018-09-26 10:39:01 +02:00
}
2015-05-11 12:29:06 -07:00
export default Tab