bootstrap-italia
Version:
Bootstrap Italia è un tema Bootstrap 5 per la creazione di applicazioni web nel pieno rispetto delle linee guida di design per i siti internet e i servizi digitali della PA
104 lines (83 loc) • 3.04 kB
JavaScript
/**
* --------------------------------------------------------------------------
* Bootstrap Italia (https://italia.github.io/bootstrap-italia/)
* Authors: https://github.com/italia/bootstrap-italia/blob/main/AUTHORS
* Licensed under BSD-3-Clause license (https://github.com/italia/bootstrap-italia/blob/main/LICENSE)
* --------------------------------------------------------------------------
*/
import BaseComponent from './base-component.js'
import EventHandler from './dom/event-handler'
import SelectorEngine from './dom/selector-engine'
import { focusSimbling } from './util/dom'
const NAME = 'accordion'
const DATA_KEY = 'bs.accordion'
const EVENT_KEY = `.${DATA_KEY}`
const DATA_API_KEY = '.data-api'
const EVENT_KEYDOWN = `keydown${EVENT_KEY}`
const EVENT_KEYDOWN_DATA_API = `keydown${EVENT_KEY}${DATA_API_KEY}`
const SELECTOR_HEADBTN_WRAPPER = '.accordion'
const SELECTOR_HEADBTN = '.accordion-item > .accordion-header [data-bs-toggle="collapse"]'
class Accordion extends BaseComponent {
constructor(element) {
super(element)
this._bindEvents()
}
// Getters
static get NAME() {
return NAME
}
// Public
handleKeyDown(keyName, target, evt) {
const eventKeyCallback = {
ArrowDown: (target) => this._focusNext(target),
ArrowUp: (target) => this._focusPrev(target),
Home: (target) => this._focusFirst(target),
End: (target) => this._focusLast(target),
}
if (typeof eventKeyCallback[keyName] === 'function') {
if (evt) {
evt.preventDefault()
}
eventKeyCallback[keyName](target)
}
}
// Private
_bindEvents() {
//fix collapse accessibility (arrow keys, home key, end key functionality)
SelectorEngine.find(SELECTOR_HEADBTN, this._element).forEach((accHead) => {
EventHandler.on(accHead, EVENT_KEYDOWN, (evt) => {
this.handleKeyDown(evt.key, evt.currentTarget, evt)
})
})
}
_getHeadButtons() {
return SelectorEngine.find(':scope > ' + SELECTOR_HEADBTN, this._element)
}
_focusNext(target) {
focusSimbling(target, this._getHeadButtons(), { loop: true })
}
_focusPrev(target) {
focusSimbling(target, this._getHeadButtons(), { isDirectionTop: true, loop: true })
}
_focusFirst(target) {
focusSimbling(target, this._getHeadButtons(), { isDirectionTop: true, isLimit: true })
}
_focusLast(target) {
focusSimbling(target, this._getHeadButtons(), { isLimit: true })
}
}
/**
* ------------------------------------------------------------------------
* Data Api implementation
* ------------------------------------------------------------------------
*/
if (typeof window !== 'undefined' && typeof document !== 'undefined') {
EventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_HEADBTN, function (evt) {
const parent = this.closest(SELECTOR_HEADBTN_WRAPPER)
if (parent) {
const accordion = Accordion.getOrCreateInstance(parent)
accordion.handleKeyDown(evt.key, this, evt)
}
})
}
export default Accordion