mirror of
https://github.com/twbs/bootstrap.git
synced 2025-02-26 23:54:23 +01:00
rewritten tab without jquery
This commit is contained in:
parent
90261b484c
commit
7f08061eca
@ -1,3 +1,5 @@
|
|||||||
|
import Util from '../util'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
* Bootstrap (v4.0.0-beta): dom/selectorEngine.js
|
* Bootstrap (v4.0.0-beta): dom/selectorEngine.js
|
||||||
@ -27,17 +29,13 @@ const SelectorEngine = (() => {
|
|||||||
if (!Element.prototype.closest) {
|
if (!Element.prototype.closest) {
|
||||||
fnClosest = (element, selector) => {
|
fnClosest = (element, selector) => {
|
||||||
let ancestor = element
|
let ancestor = element
|
||||||
if (!document.documentElement.contains(element)) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
do {
|
do {
|
||||||
if (fnMatches.call(ancestor, selector)) {
|
if (fnMatches.call(ancestor, selector)) {
|
||||||
return ancestor
|
return ancestor
|
||||||
}
|
}
|
||||||
|
|
||||||
ancestor = ancestor.parentElement
|
ancestor = ancestor.parentElement
|
||||||
} while (ancestor !== null)
|
} while (ancestor !== null && ancestor.nodeType === Node.ELEMENT_NODE)
|
||||||
|
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
@ -48,12 +46,67 @@ const SelectorEngine = (() => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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 {
|
return {
|
||||||
matches(element, selector) {
|
matches(element, selector) {
|
||||||
return fnMatches.call(element, selector)
|
return fnMatches.call(element, selector)
|
||||||
},
|
},
|
||||||
|
|
||||||
find(selector, element = document) {
|
find(selector, element = document.documentElement) {
|
||||||
if (typeof selector !== 'string') {
|
if (typeof selector !== 'string') {
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
@ -62,21 +115,24 @@ const SelectorEngine = (() => {
|
|||||||
return SelectorEngine.findOne(selector, element)
|
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') {
|
if (typeof selector !== 'string') {
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
let selectorType = 'querySelector'
|
return findOneFn.call(element, selector)
|
||||||
if (selector.indexOf('#') === 0) {
|
},
|
||||||
selectorType = 'getElementById'
|
|
||||||
selector = selector.substr(1, selector.length)
|
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) {
|
closest(element, selector) {
|
||||||
|
120
js/src/tab.js
120
js/src/tab.js
@ -5,7 +5,9 @@
|
|||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import $ from 'jquery'
|
import Data from './dom/data'
|
||||||
|
import EventHandler from './dom/eventHandler'
|
||||||
|
import SelectorEngine from './dom/selectorEngine'
|
||||||
import Util from './util'
|
import Util from './util'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -19,7 +21,6 @@ const VERSION = '4.3.1'
|
|||||||
const DATA_KEY = 'bs.tab'
|
const DATA_KEY = 'bs.tab'
|
||||||
const EVENT_KEY = `.${DATA_KEY}`
|
const EVENT_KEY = `.${DATA_KEY}`
|
||||||
const DATA_API_KEY = '.data-api'
|
const DATA_API_KEY = '.data-api'
|
||||||
const JQUERY_NO_CONFLICT = $.fn[NAME]
|
|
||||||
|
|
||||||
const Event = {
|
const Event = {
|
||||||
HIDE : `hide${EVENT_KEY}`,
|
HIDE : `hide${EVENT_KEY}`,
|
||||||
@ -41,10 +42,10 @@ const Selector = {
|
|||||||
DROPDOWN : '.dropdown',
|
DROPDOWN : '.dropdown',
|
||||||
NAV_LIST_GROUP : '.nav, .list-group',
|
NAV_LIST_GROUP : '.nav, .list-group',
|
||||||
ACTIVE : '.active',
|
ACTIVE : '.active',
|
||||||
ACTIVE_UL : '> li > .active',
|
ACTIVE_UL : ':scope > li > .active',
|
||||||
DATA_TOGGLE : '[data-toggle="tab"], [data-toggle="pill"], [data-toggle="list"]',
|
DATA_TOGGLE : '[data-toggle="tab"], [data-toggle="pill"], [data-toggle="list"]',
|
||||||
DROPDOWN_TOGGLE : '.dropdown-toggle',
|
DROPDOWN_TOGGLE : '.dropdown-toggle',
|
||||||
DROPDOWN_ACTIVE_CHILD : '> .dropdown-menu .active'
|
DROPDOWN_ACTIVE_CHILD : ':scope > .dropdown-menu .active'
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -56,6 +57,8 @@ const Selector = {
|
|||||||
class Tab {
|
class Tab {
|
||||||
constructor(element) {
|
constructor(element) {
|
||||||
this._element = element
|
this._element = element
|
||||||
|
|
||||||
|
Data.setData(this._element, DATA_KEY, this)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Getters
|
// Getters
|
||||||
@ -68,39 +71,37 @@ class Tab {
|
|||||||
|
|
||||||
show() {
|
show() {
|
||||||
if (this._element.parentNode &&
|
if (this._element.parentNode &&
|
||||||
this._element.parentNode.nodeType === Node.ELEMENT_NODE &&
|
this._element.parentNode.nodeType === Node.ELEMENT_NODE &&
|
||||||
$(this._element).hasClass(ClassName.ACTIVE) ||
|
this._element.classList.contains(ClassName.ACTIVE) ||
|
||||||
$(this._element).hasClass(ClassName.DISABLED)) {
|
this._element.classList.contains(ClassName.DISABLED)) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
let target
|
let target
|
||||||
let previous
|
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)
|
const selector = Util.getSelectorFromElement(this._element)
|
||||||
|
|
||||||
if (listElement) {
|
if (listElement) {
|
||||||
const itemSelector = listElement.nodeName === 'UL' || listElement.nodeName === 'OL' ? Selector.ACTIVE_UL : Selector.ACTIVE
|
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]
|
previous = previous[previous.length - 1]
|
||||||
}
|
}
|
||||||
|
|
||||||
const hideEvent = $.Event(Event.HIDE, {
|
let hideEvent = null
|
||||||
relatedTarget: this._element
|
|
||||||
})
|
|
||||||
|
|
||||||
const showEvent = $.Event(Event.SHOW, {
|
if (previous) {
|
||||||
|
hideEvent = EventHandler.trigger(previous, Event.HIDE, {
|
||||||
|
relatedTarget: this._element
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const showEvent = EventHandler.trigger(this._element, Event.SHOW, {
|
||||||
relatedTarget: previous
|
relatedTarget: previous
|
||||||
})
|
})
|
||||||
|
|
||||||
if (previous) {
|
if (showEvent.defaultPrevented ||
|
||||||
$(previous).trigger(hideEvent)
|
hideEvent !== null && hideEvent.defaultPrevented) {
|
||||||
}
|
|
||||||
|
|
||||||
$(this._element).trigger(showEvent)
|
|
||||||
|
|
||||||
if (showEvent.isDefaultPrevented() ||
|
|
||||||
hideEvent.isDefaultPrevented()) {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -114,16 +115,12 @@ class Tab {
|
|||||||
)
|
)
|
||||||
|
|
||||||
const complete = () => {
|
const complete = () => {
|
||||||
const hiddenEvent = $.Event(Event.HIDDEN, {
|
EventHandler.trigger(previous, Event.HIDDEN, {
|
||||||
relatedTarget: this._element
|
relatedTarget: this._element
|
||||||
})
|
})
|
||||||
|
EventHandler.trigger(this._element, Event.SHOWN, {
|
||||||
const shownEvent = $.Event(Event.SHOWN, {
|
|
||||||
relatedTarget: previous
|
relatedTarget: previous
|
||||||
})
|
})
|
||||||
|
|
||||||
$(previous).trigger(hiddenEvent)
|
|
||||||
$(this._element).trigger(shownEvent)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (target) {
|
if (target) {
|
||||||
@ -134,7 +131,7 @@ class Tab {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dispose() {
|
dispose() {
|
||||||
$.removeData(this._element, DATA_KEY)
|
Data.removeData(this._element, DATA_KEY)
|
||||||
this._element = null
|
this._element = null
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -142,11 +139,13 @@ class Tab {
|
|||||||
|
|
||||||
_activate(element, container, callback) {
|
_activate(element, container, callback) {
|
||||||
const activeElements = container && (container.nodeName === 'UL' || container.nodeName === 'OL')
|
const activeElements = container && (container.nodeName === 'UL' || container.nodeName === 'OL')
|
||||||
? $(container).find(Selector.ACTIVE_UL)
|
? SelectorEngine.find(Selector.ACTIVE_UL, container)
|
||||||
: $(container).children(Selector.ACTIVE)
|
: 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(
|
const complete = () => this._transitionComplete(
|
||||||
element,
|
element,
|
||||||
active,
|
active,
|
||||||
@ -155,10 +154,9 @@ class Tab {
|
|||||||
|
|
||||||
if (active && isTransitioning) {
|
if (active && isTransitioning) {
|
||||||
const transitionDuration = Util.getTransitionDurationFromElement(active)
|
const transitionDuration = Util.getTransitionDurationFromElement(active)
|
||||||
|
active.classList.remove(ClassName.SHOW)
|
||||||
|
|
||||||
$(active)
|
EventHandler.one(active, Util.TRANSITION_END, complete)
|
||||||
.removeClass(ClassName.SHOW)
|
|
||||||
.one(Util.TRANSITION_END, complete)
|
|
||||||
Util.emulateTransitionEnd(active, transitionDuration)
|
Util.emulateTransitionEnd(active, transitionDuration)
|
||||||
} else {
|
} else {
|
||||||
complete()
|
complete()
|
||||||
@ -167,14 +165,12 @@ class Tab {
|
|||||||
|
|
||||||
_transitionComplete(element, active, callback) {
|
_transitionComplete(element, active, callback) {
|
||||||
if (active) {
|
if (active) {
|
||||||
$(active).removeClass(ClassName.ACTIVE)
|
active.classList.remove(ClassName.ACTIVE)
|
||||||
|
|
||||||
const dropdownChild = $(active.parentNode).find(
|
const dropdownChild = SelectorEngine.findOne(Selector.DROPDOWN_ACTIVE_CHILD, active.parentNode)
|
||||||
Selector.DROPDOWN_ACTIVE_CHILD
|
|
||||||
)[0]
|
|
||||||
|
|
||||||
if (dropdownChild) {
|
if (dropdownChild) {
|
||||||
$(dropdownChild).removeClass(ClassName.ACTIVE)
|
dropdownChild.classList.remove(ClassName.ACTIVE)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (active.getAttribute('role') === 'tab') {
|
if (active.getAttribute('role') === 'tab') {
|
||||||
@ -182,7 +178,7 @@ class Tab {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$(element).addClass(ClassName.ACTIVE)
|
element.classList.add(ClassName.ACTIVE)
|
||||||
if (element.getAttribute('role') === 'tab') {
|
if (element.getAttribute('role') === 'tab') {
|
||||||
element.setAttribute('aria-selected', true)
|
element.setAttribute('aria-selected', true)
|
||||||
}
|
}
|
||||||
@ -193,13 +189,12 @@ class Tab {
|
|||||||
element.classList.add(ClassName.SHOW)
|
element.classList.add(ClassName.SHOW)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (element.parentNode && $(element.parentNode).hasClass(ClassName.DROPDOWN_MENU)) {
|
if (element.parentNode && element.parentNode.classList.contains(ClassName.DROPDOWN_MENU)) {
|
||||||
const dropdownElement = $(element).closest(Selector.DROPDOWN)[0]
|
const dropdownElement = SelectorEngine.closest(element, Selector.DROPDOWN)
|
||||||
|
|
||||||
if (dropdownElement) {
|
if (dropdownElement) {
|
||||||
const dropdownToggleList = [].slice.call(dropdownElement.querySelectorAll(Selector.DROPDOWN_TOGGLE))
|
Util.makeArray(dropdownElement.querySelectorAll(Selector.DROPDOWN_TOGGLE))
|
||||||
|
.forEach((dropdown) => dropdown.classList.add(ClassName.ACTIVE))
|
||||||
$(dropdownToggleList).addClass(ClassName.ACTIVE)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
element.setAttribute('aria-expanded', true)
|
element.setAttribute('aria-expanded', true)
|
||||||
@ -214,13 +209,7 @@ class Tab {
|
|||||||
|
|
||||||
static _jQueryInterface(config) {
|
static _jQueryInterface(config) {
|
||||||
return this.each(function () {
|
return this.each(function () {
|
||||||
const $this = $(this)
|
const data = Data.getData(this, DATA_KEY) || new Tab(this)
|
||||||
let data = $this.data(DATA_KEY)
|
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
data = new Tab(this)
|
|
||||||
$this.data(DATA_KEY, data)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof config === 'string') {
|
if (typeof config === 'string') {
|
||||||
if (typeof data[config] === 'undefined') {
|
if (typeof data[config] === 'undefined') {
|
||||||
@ -238,11 +227,12 @@ class Tab {
|
|||||||
* ------------------------------------------------------------------------
|
* ------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
$(document)
|
EventHandler.on(document, Event.CLICK_DATA_API, Selector.DATA_TOGGLE, function (event) {
|
||||||
.on(Event.CLICK_DATA_API, Selector.DATA_TOGGLE, function (event) {
|
event.preventDefault()
|
||||||
event.preventDefault()
|
|
||||||
Tab._jQueryInterface.call($(this), 'show')
|
const data = Data.getData(this, DATA_KEY) || new Tab(this)
|
||||||
})
|
data.show()
|
||||||
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ------------------------------------------------------------------------
|
* ------------------------------------------------------------------------
|
||||||
@ -250,11 +240,15 @@ $(document)
|
|||||||
* ------------------------------------------------------------------------
|
* ------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
$.fn[NAME] = Tab._jQueryInterface
|
const $ = Util.jQuery
|
||||||
$.fn[NAME].Constructor = Tab
|
if (typeof $ !== 'undefined') {
|
||||||
$.fn[NAME].noConflict = () => {
|
const JQUERY_NO_CONFLICT = $.fn[NAME]
|
||||||
$.fn[NAME] = JQUERY_NO_CONFLICT
|
$.fn[NAME] = Tab._jQueryInterface
|
||||||
return Tab._jQueryInterface
|
$.fn[NAME].Constructor = Tab
|
||||||
|
$.fn[NAME].noConflict = () => {
|
||||||
|
$.fn[NAME] = JQUERY_NO_CONFLICT
|
||||||
|
return Tab._jQueryInterface
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Tab
|
export default Tab
|
||||||
|
@ -320,7 +320,7 @@ $(function () {
|
|||||||
'</ul>'
|
'</ul>'
|
||||||
var $tabs = $(tabsHTML).appendTo('#qunit-fixture')
|
var $tabs = $(tabsHTML).appendTo('#qunit-fixture')
|
||||||
|
|
||||||
$tabs.find('li:last-child a').trigger('click')
|
EventHandler.trigger($tabs.find('li:last-child a')[0], 'click')
|
||||||
assert.notOk($tabs.find('li:first-child a').hasClass('active'))
|
assert.notOk($tabs.find('li:first-child a').hasClass('active'))
|
||||||
assert.ok($tabs.find('li:last-child a').hasClass('active'))
|
assert.ok($tabs.find('li:last-child a').hasClass('active'))
|
||||||
})
|
})
|
||||||
@ -339,7 +339,7 @@ $(function () {
|
|||||||
'</ul>'
|
'</ul>'
|
||||||
var $tabs = $(tabsHTML).appendTo('#qunit-fixture')
|
var $tabs = $(tabsHTML).appendTo('#qunit-fixture')
|
||||||
|
|
||||||
$tabs.find('li:first-child a').trigger('click')
|
EventHandler.trigger($tabs.find('li:first-child a')[0], 'click')
|
||||||
assert.ok($tabs.find('li:first-child a').hasClass('active'))
|
assert.ok($tabs.find('li:first-child a').hasClass('active'))
|
||||||
assert.notOk($tabs.find('li:last-child a').hasClass('active'))
|
assert.notOk($tabs.find('li:last-child a').hasClass('active'))
|
||||||
assert.notOk($tabs.find('li:last-child .dropdown-menu a:first-child').hasClass('active'))
|
assert.notOk($tabs.find('li:last-child .dropdown-menu a:first-child').hasClass('active'))
|
||||||
@ -378,9 +378,10 @@ $(function () {
|
|||||||
|
|
||||||
$('#tab1').on('shown.bs.tab', function () {
|
$('#tab1').on('shown.bs.tab', function () {
|
||||||
assert.ok($('#x-tab1').hasClass('active'))
|
assert.ok($('#x-tab1').hasClass('active'))
|
||||||
$('#tabNested2').trigger($.Event('click'))
|
EventHandler.trigger($('#tabNested2')[0], 'click')
|
||||||
})
|
})
|
||||||
.trigger($.Event('click'))
|
|
||||||
|
EventHandler.trigger($('#tab1')[0], 'click')
|
||||||
})
|
})
|
||||||
|
|
||||||
QUnit.test('should not remove fade class if no active pane is present', function (assert) {
|
QUnit.test('should not remove fade class if no active pane is present', function (assert) {
|
||||||
@ -410,9 +411,11 @@ $(function () {
|
|||||||
|
|
||||||
done()
|
done()
|
||||||
})
|
})
|
||||||
.trigger($.Event('click'))
|
|
||||||
|
EventHandler.trigger($('#tab-home')[0], 'click')
|
||||||
})
|
})
|
||||||
.trigger($.Event('click'))
|
|
||||||
|
EventHandler.trigger($('#tab-profile')[0], 'click')
|
||||||
})
|
})
|
||||||
|
|
||||||
QUnit.test('should handle removed tabs', function (assert) {
|
QUnit.test('should handle removed tabs', function (assert) {
|
||||||
|
@ -227,7 +227,10 @@
|
|||||||
|
|
||||||
<script src="../../../node_modules/jquery/dist/jquery.slim.min.js"></script>
|
<script src="../../../node_modules/jquery/dist/jquery.slim.min.js"></script>
|
||||||
<script src="../../../node_modules/popper.js/dist/umd/popper.min.js"></script>
|
<script src="../../../node_modules/popper.js/dist/umd/popper.min.js"></script>
|
||||||
|
<script src="../../dist/dom/data.js"></script>
|
||||||
<script src="../../dist/dom/eventHandler.js"></script>
|
<script src="../../dist/dom/eventHandler.js"></script>
|
||||||
|
<script src="../../dist/dom/manipulator.js"></script>
|
||||||
|
<script src="../../dist/dom/selectorEngine.js"></script>
|
||||||
<script src="../../dist/util.js"></script>
|
<script src="../../dist/util.js"></script>
|
||||||
<script src="../../dist/tab.js"></script>
|
<script src="../../dist/tab.js"></script>
|
||||||
<script src="../../dist/dropdown.js"></script>
|
<script src="../../dist/dropdown.js"></script>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user