+import Util from '../util'
+
/**
* --------------------------------------------------------------------------
* Bootstrap (v4.0.0-beta): dom/selectorEngine.js
if (!Element.prototype.closest) {
fnClosest = (element, selector) => {
let ancestor = element
- if (!document.documentElement.contains(element)) {
- return null
- }
-
do {
if (fnMatches.call(ancestor, selector)) {
return ancestor
}
ancestor = ancestor.parentElement
- } while (ancestor !== null)
+ } while (ancestor !== null && ancestor.nodeType === Node.ELEMENT_NODE)
return null
}
}
}
+ const scopeSelectorRegex = /:scope\b/
+ const supportScopeQuery = (() => {
+ const element = document.createElement('div')
+ try {
+ element.querySelectorAll(':scope *')
+ } catch (e) {
+ return false
+ }
+
+ return true
+ })()
+
+ let findFn = null
+ let findOneFn = null
+ if (supportScopeQuery) {
+ findFn = Element.prototype.querySelectorAll
+ findOneFn = Element.prototype.querySelector
+ } else {
+ findFn = function (selector) {
+ if (!scopeSelectorRegex.test(selector)) {
+ return this.querySelectorAll(selector)
+ }
+
+ const hasId = Boolean(this.id)
+ if (!hasId) {
+ this.id = Util.getUID('scope')
+ }
+
+ let nodeList = null
+ try {
+ selector = selector.replace(scopeSelectorRegex, `#${this.id}`)
+ nodeList = this.querySelectorAll(selector)
+ } finally {
+ if (!hasId) {
+ this.removeAttribute('id')
+ }
+ }
+
+ return nodeList
+ }
+
+ findOneFn = function (selector) {
+ if (!scopeSelectorRegex.test(selector)) {
+ return this.querySelector(selector)
+ }
+
+ const matches = findFn.call(this, selector)
+ if (typeof matches[0] !== 'undefined') {
+ return matches[0]
+ }
+
+ return null
+ }
+ }
+
return {
matches(element, selector) {
return fnMatches.call(element, selector)
},
- find(selector, element = document) {
+ find(selector, element = document.documentElement) {
if (typeof selector !== 'string') {
return null
}
return SelectorEngine.findOne(selector, element)
}
- return element.querySelectorAll(selector)
+ return findFn.call(element, selector)
},
- findOne(selector, element = document) {
+ findOne(selector, element = document.documentElement) {
if (typeof selector !== 'string') {
return null
}
- let selectorType = 'querySelector'
- if (selector.indexOf('#') === 0) {
- selectorType = 'getElementById'
- selector = selector.substr(1, selector.length)
+ return findOneFn.call(element, selector)
+ },
+
+ children(element, selector) {
+ if (typeof selector !== 'string') {
+ return null
}
- return element[selectorType](selector)
+ const children = Util.makeArray(element.children)
+ return children.filter((child) => this.matches(child, selector))
},
closest(element, selector) {
* --------------------------------------------------------------------------
*/
-import $ from 'jquery'
+import Data from './dom/data'
+import EventHandler from './dom/eventHandler'
+import SelectorEngine from './dom/selectorEngine'
import Util from './util'
/**
const DATA_KEY = 'bs.tab'
const EVENT_KEY = `.${DATA_KEY}`
const DATA_API_KEY = '.data-api'
-const JQUERY_NO_CONFLICT = $.fn[NAME]
const Event = {
HIDE : `hide${EVENT_KEY}`,
DROPDOWN : '.dropdown',
NAV_LIST_GROUP : '.nav, .list-group',
ACTIVE : '.active',
- ACTIVE_UL : '> li > .active',
+ ACTIVE_UL : ':scope > li > .active',
DATA_TOGGLE : '[data-toggle="tab"], [data-toggle="pill"], [data-toggle="list"]',
DROPDOWN_TOGGLE : '.dropdown-toggle',
- DROPDOWN_ACTIVE_CHILD : '> .dropdown-menu .active'
+ DROPDOWN_ACTIVE_CHILD : ':scope > .dropdown-menu .active'
}
/**
class Tab {
constructor(element) {
this._element = element
+
+ Data.setData(this._element, DATA_KEY, this)
}
// Getters
show() {
if (this._element.parentNode &&
- this._element.parentNode.nodeType === Node.ELEMENT_NODE &&
- $(this._element).hasClass(ClassName.ACTIVE) ||
- $(this._element).hasClass(ClassName.DISABLED)) {
+ this._element.parentNode.nodeType === Node.ELEMENT_NODE &&
+ this._element.classList.contains(ClassName.ACTIVE) ||
+ this._element.classList.contains(ClassName.DISABLED)) {
return
}
let target
let previous
- const listElement = $(this._element).closest(Selector.NAV_LIST_GROUP)[0]
+ const listElement = SelectorEngine.closest(this._element, Selector.NAV_LIST_GROUP)
const selector = Util.getSelectorFromElement(this._element)
if (listElement) {
const itemSelector = listElement.nodeName === 'UL' || listElement.nodeName === 'OL' ? Selector.ACTIVE_UL : Selector.ACTIVE
- previous = $.makeArray($(listElement).find(itemSelector))
+ previous = Util.makeArray(SelectorEngine.find(itemSelector, listElement))
previous = previous[previous.length - 1]
}
- const hideEvent = $.Event(Event.HIDE, {
- relatedTarget: this._element
- })
-
- const showEvent = $.Event(Event.SHOW, {
- relatedTarget: previous
- })
+ let hideEvent = null
if (previous) {
- $(previous).trigger(hideEvent)
+ hideEvent = EventHandler.trigger(previous, Event.HIDE, {
+ relatedTarget: this._element
+ })
}
- $(this._element).trigger(showEvent)
+ const showEvent = EventHandler.trigger(this._element, Event.SHOW, {
+ relatedTarget: previous
+ })
- if (showEvent.isDefaultPrevented() ||
- hideEvent.isDefaultPrevented()) {
+ if (showEvent.defaultPrevented ||
+ hideEvent !== null && hideEvent.defaultPrevented) {
return
}
)
const complete = () => {
- const hiddenEvent = $.Event(Event.HIDDEN, {
+ EventHandler.trigger(previous, Event.HIDDEN, {
relatedTarget: this._element
})
-
- const shownEvent = $.Event(Event.SHOWN, {
+ EventHandler.trigger(this._element, Event.SHOWN, {
relatedTarget: previous
})
-
- $(previous).trigger(hiddenEvent)
- $(this._element).trigger(shownEvent)
}
if (target) {
}
dispose() {
- $.removeData(this._element, DATA_KEY)
+ Data.removeData(this._element, DATA_KEY)
this._element = null
}
_activate(element, container, callback) {
const activeElements = container && (container.nodeName === 'UL' || container.nodeName === 'OL')
- ? $(container).find(Selector.ACTIVE_UL)
- : $(container).children(Selector.ACTIVE)
+ ? SelectorEngine.find(Selector.ACTIVE_UL, container)
+ : SelectorEngine.children(container, Selector.ACTIVE)
+
+ const active = activeElements[0]
+ const isTransitioning = callback &&
+ (active && active.classList.contains(ClassName.FADE))
- const active = activeElements[0]
- const isTransitioning = callback && (active && $(active).hasClass(ClassName.FADE))
const complete = () => this._transitionComplete(
element,
active,
if (active && isTransitioning) {
const transitionDuration = Util.getTransitionDurationFromElement(active)
+ active.classList.remove(ClassName.SHOW)
- $(active)
- .removeClass(ClassName.SHOW)
- .one(Util.TRANSITION_END, complete)
+ EventHandler.one(active, Util.TRANSITION_END, complete)
Util.emulateTransitionEnd(active, transitionDuration)
} else {
complete()
_transitionComplete(element, active, callback) {
if (active) {
- $(active).removeClass(ClassName.ACTIVE)
+ active.classList.remove(ClassName.ACTIVE)
- const dropdownChild = $(active.parentNode).find(
- Selector.DROPDOWN_ACTIVE_CHILD
- )[0]
+ const dropdownChild = SelectorEngine.findOne(Selector.DROPDOWN_ACTIVE_CHILD, active.parentNode)
if (dropdownChild) {
- $(dropdownChild).removeClass(ClassName.ACTIVE)
+ dropdownChild.classList.remove(ClassName.ACTIVE)
}
if (active.getAttribute('role') === 'tab') {
}
}
- $(element).addClass(ClassName.ACTIVE)
+ element.classList.add(ClassName.ACTIVE)
if (element.getAttribute('role') === 'tab') {
element.setAttribute('aria-selected', true)
}
element.classList.add(ClassName.SHOW)
}
- if (element.parentNode && $(element.parentNode).hasClass(ClassName.DROPDOWN_MENU)) {
- const dropdownElement = $(element).closest(Selector.DROPDOWN)[0]
+ if (element.parentNode && element.parentNode.classList.contains(ClassName.DROPDOWN_MENU)) {
+ const dropdownElement = SelectorEngine.closest(element, Selector.DROPDOWN)
if (dropdownElement) {
- const dropdownToggleList = [].slice.call(dropdownElement.querySelectorAll(Selector.DROPDOWN_TOGGLE))
-
- $(dropdownToggleList).addClass(ClassName.ACTIVE)
+ Util.makeArray(dropdownElement.querySelectorAll(Selector.DROPDOWN_TOGGLE))
+ .forEach((dropdown) => dropdown.classList.add(ClassName.ACTIVE))
}
element.setAttribute('aria-expanded', true)
static _jQueryInterface(config) {
return this.each(function () {
- const $this = $(this)
- let data = $this.data(DATA_KEY)
-
- if (!data) {
- data = new Tab(this)
- $this.data(DATA_KEY, data)
- }
+ const data = Data.getData(this, DATA_KEY) || new Tab(this)
if (typeof config === 'string') {
if (typeof data[config] === 'undefined') {
* ------------------------------------------------------------------------
*/
-$(document)
- .on(Event.CLICK_DATA_API, Selector.DATA_TOGGLE, function (event) {
- event.preventDefault()
- Tab._jQueryInterface.call($(this), 'show')
- })
+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()
+})
/**
* ------------------------------------------------------------------------
* ------------------------------------------------------------------------
*/
-$.fn[NAME] = Tab._jQueryInterface
-$.fn[NAME].Constructor = Tab
-$.fn[NAME].noConflict = () => {
- $.fn[NAME] = JQUERY_NO_CONFLICT
- return Tab._jQueryInterface
+const $ = Util.jQuery
+if (typeof $ !== 'undefined') {
+ const JQUERY_NO_CONFLICT = $.fn[NAME]
+ $.fn[NAME] = Tab._jQueryInterface
+ $.fn[NAME].Constructor = Tab
+ $.fn[NAME].noConflict = () => {
+ $.fn[NAME] = JQUERY_NO_CONFLICT
+ return Tab._jQueryInterface
+ }
}
export default Tab