Spaces:
Running
Running
| /** | |
| * -------------------------------------------------------------------------- | |
| * Bootstrap (v5.1.3): tooltip.js | |
| * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) | |
| * -------------------------------------------------------------------------- | |
| */ | |
| import * as Popper from '@popperjs/core' | |
| import { | |
| defineJQueryPlugin, | |
| findShadowRoot, | |
| getElement, | |
| getUID, | |
| isElement, | |
| isRTL, | |
| noop, | |
| typeCheckConfig | |
| } from './util/index' | |
| import { | |
| DefaultAllowlist, | |
| sanitizeHtml | |
| } from './util/sanitizer' | |
| import Data from './dom/data' | |
| import EventHandler from './dom/event-handler' | |
| import Manipulator from './dom/manipulator' | |
| import SelectorEngine from './dom/selector-engine' | |
| import BaseComponent from './base-component' | |
| /** | |
| * ------------------------------------------------------------------------ | |
| * Constants | |
| * ------------------------------------------------------------------------ | |
| */ | |
| const NAME = 'tooltip' | |
| const DATA_KEY = 'bs.tooltip' | |
| const EVENT_KEY = `.${DATA_KEY}` | |
| const CLASS_PREFIX = 'bs-tooltip' | |
| const DISALLOWED_ATTRIBUTES = new Set(['sanitize', 'allowList', 'sanitizeFn']) | |
| const DefaultType = { | |
| animation: 'boolean', | |
| template: 'string', | |
| title: '(string|element|function)', | |
| trigger: 'string', | |
| delay: '(number|object)', | |
| html: 'boolean', | |
| selector: '(string|boolean)', | |
| placement: '(string|function)', | |
| offset: '(array|string|function)', | |
| container: '(string|element|boolean)', | |
| fallbackPlacements: 'array', | |
| boundary: '(string|element)', | |
| customClass: '(string|function)', | |
| sanitize: 'boolean', | |
| sanitizeFn: '(null|function)', | |
| allowList: 'object', | |
| popperConfig: '(null|object|function)' | |
| } | |
| const AttachmentMap = { | |
| AUTO: 'auto', | |
| TOP: 'top', | |
| RIGHT: isRTL() ? 'left' : 'right', | |
| BOTTOM: 'bottom', | |
| LEFT: isRTL() ? 'right' : 'left' | |
| } | |
| const Default = { | |
| animation: true, | |
| template: '<div class="tooltip" role="tooltip">' + | |
| '<div class="tooltip-arrow"></div>' + | |
| '<div class="tooltip-inner"></div>' + | |
| '</div>', | |
| trigger: 'hover focus', | |
| title: '', | |
| delay: 0, | |
| html: false, | |
| selector: false, | |
| placement: 'top', | |
| offset: [0, 0], | |
| container: false, | |
| fallbackPlacements: ['top', 'right', 'bottom', 'left'], | |
| boundary: 'clippingParents', | |
| customClass: '', | |
| sanitize: true, | |
| sanitizeFn: null, | |
| allowList: DefaultAllowlist, | |
| popperConfig: null | |
| } | |
| const Event = { | |
| HIDE: `hide${EVENT_KEY}`, | |
| HIDDEN: `hidden${EVENT_KEY}`, | |
| SHOW: `show${EVENT_KEY}`, | |
| SHOWN: `shown${EVENT_KEY}`, | |
| INSERTED: `inserted${EVENT_KEY}`, | |
| CLICK: `click${EVENT_KEY}`, | |
| FOCUSIN: `focusin${EVENT_KEY}`, | |
| FOCUSOUT: `focusout${EVENT_KEY}`, | |
| MOUSEENTER: `mouseenter${EVENT_KEY}`, | |
| MOUSELEAVE: `mouseleave${EVENT_KEY}` | |
| } | |
| const CLASS_NAME_FADE = 'fade' | |
| const CLASS_NAME_MODAL = 'modal' | |
| const CLASS_NAME_SHOW = 'show' | |
| const HOVER_STATE_SHOW = 'show' | |
| const HOVER_STATE_OUT = 'out' | |
| const SELECTOR_TOOLTIP_INNER = '.tooltip-inner' | |
| const SELECTOR_MODAL = `.${CLASS_NAME_MODAL}` | |
| const EVENT_MODAL_HIDE = 'hide.bs.modal' | |
| const TRIGGER_HOVER = 'hover' | |
| const TRIGGER_FOCUS = 'focus' | |
| const TRIGGER_CLICK = 'click' | |
| const TRIGGER_MANUAL = 'manual' | |
| /** | |
| * ------------------------------------------------------------------------ | |
| * Class Definition | |
| * ------------------------------------------------------------------------ | |
| */ | |
| class Tooltip extends BaseComponent { | |
| constructor(element, config) { | |
| if (typeof Popper === 'undefined') { | |
| throw new TypeError('Bootstrap\'s tooltips require Popper (https://popper.js.org)') | |
| } | |
| super(element) | |
| // private | |
| this._isEnabled = true | |
| this._timeout = 0 | |
| this._hoverState = '' | |
| this._activeTrigger = {} | |
| this._popper = null | |
| // Protected | |
| this._config = this._getConfig(config) | |
| this.tip = null | |
| this._setListeners() | |
| } | |
| // Getters | |
| static get Default() { | |
| return Default | |
| } | |
| static get NAME() { | |
| return NAME | |
| } | |
| static get Event() { | |
| return Event | |
| } | |
| static get DefaultType() { | |
| return DefaultType | |
| } | |
| // Public | |
| enable() { | |
| this._isEnabled = true | |
| } | |
| disable() { | |
| this._isEnabled = false | |
| } | |
| toggleEnabled() { | |
| this._isEnabled = !this._isEnabled | |
| } | |
| toggle(event) { | |
| if (!this._isEnabled) { | |
| return | |
| } | |
| if (event) { | |
| const context = this._initializeOnDelegatedTarget(event) | |
| context._activeTrigger.click = !context._activeTrigger.click | |
| if (context._isWithActiveTrigger()) { | |
| context._enter(null, context) | |
| } else { | |
| context._leave(null, context) | |
| } | |
| } else { | |
| if (this.getTipElement().classList.contains(CLASS_NAME_SHOW)) { | |
| this._leave(null, this) | |
| return | |
| } | |
| this._enter(null, this) | |
| } | |
| } | |
| dispose() { | |
| clearTimeout(this._timeout) | |
| EventHandler.off(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler) | |
| if (this.tip) { | |
| this.tip.remove() | |
| } | |
| this._disposePopper() | |
| super.dispose() | |
| } | |
| show() { | |
| if (this._element.style.display === 'none') { | |
| throw new Error('Please use show on visible elements') | |
| } | |
| if (!(this.isWithContent() && this._isEnabled)) { | |
| return | |
| } | |
| const showEvent = EventHandler.trigger(this._element, this.constructor.Event.SHOW) | |
| const shadowRoot = findShadowRoot(this._element) | |
| const isInTheDom = shadowRoot === null ? | |
| this._element.ownerDocument.documentElement.contains(this._element) : | |
| shadowRoot.contains(this._element) | |
| if (showEvent.defaultPrevented || !isInTheDom) { | |
| return | |
| } | |
| // A trick to recreate a tooltip in case a new title is given by using the NOT documented `data-bs-original-title` | |
| // This will be removed later in favor of a `setContent` method | |
| if (this.constructor.NAME === 'tooltip' && this.tip && this.getTitle() !== this.tip.querySelector(SELECTOR_TOOLTIP_INNER).innerHTML) { | |
| this._disposePopper() | |
| this.tip.remove() | |
| this.tip = null | |
| } | |
| const tip = this.getTipElement() | |
| const tipId = getUID(this.constructor.NAME) | |
| tip.setAttribute('id', tipId) | |
| this._element.setAttribute('aria-describedby', tipId) | |
| if (this._config.animation) { | |
| tip.classList.add(CLASS_NAME_FADE) | |
| } | |
| const placement = typeof this._config.placement === 'function' ? | |
| this._config.placement.call(this, tip, this._element) : | |
| this._config.placement | |
| const attachment = this._getAttachment(placement) | |
| this._addAttachmentClass(attachment) | |
| const { | |
| container | |
| } = this._config | |
| Data.set(tip, this.constructor.DATA_KEY, this) | |
| if (!this._element.ownerDocument.documentElement.contains(this.tip)) { | |
| container.append(tip) | |
| EventHandler.trigger(this._element, this.constructor.Event.INSERTED) | |
| } | |
| if (this._popper) { | |
| this._popper.update() | |
| } else { | |
| this._popper = Popper.createPopper(this._element, tip, this._getPopperConfig(attachment)) | |
| } | |
| tip.classList.add(CLASS_NAME_SHOW) | |
| const customClass = this._resolvePossibleFunction(this._config.customClass) | |
| if (customClass) { | |
| tip.classList.add(...customClass.split(' ')) | |
| } | |
| // If this is a touch-enabled device we add extra | |
| // empty mouseover listeners to the body's immediate children; | |
| // only needed because of broken event delegation on iOS | |
| // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html | |
| if ('ontouchstart' in document.documentElement) { | |
| [].concat(...document.body.children).forEach(element => { | |
| EventHandler.on(element, 'mouseover', noop) | |
| }) | |
| } | |
| const complete = () => { | |
| const prevHoverState = this._hoverState | |
| this._hoverState = null | |
| EventHandler.trigger(this._element, this.constructor.Event.SHOWN) | |
| if (prevHoverState === HOVER_STATE_OUT) { | |
| this._leave(null, this) | |
| } | |
| } | |
| const isAnimated = this.tip.classList.contains(CLASS_NAME_FADE) | |
| this._queueCallback(complete, this.tip, isAnimated) | |
| } | |
| hide() { | |
| if (!this._popper) { | |
| return | |
| } | |
| const tip = this.getTipElement() | |
| const complete = () => { | |
| if (this._isWithActiveTrigger()) { | |
| return | |
| } | |
| if (this._hoverState !== HOVER_STATE_SHOW) { | |
| tip.remove() | |
| } | |
| this._cleanTipClass() | |
| this._element.removeAttribute('aria-describedby') | |
| EventHandler.trigger(this._element, this.constructor.Event.HIDDEN) | |
| this._disposePopper() | |
| } | |
| const hideEvent = EventHandler.trigger(this._element, this.constructor.Event.HIDE) | |
| if (hideEvent.defaultPrevented) { | |
| return | |
| } | |
| tip.classList.remove(CLASS_NAME_SHOW) | |
| // If this is a touch-enabled device we remove the extra | |
| // empty mouseover listeners we added for iOS support | |
| if ('ontouchstart' in document.documentElement) { | |
| [].concat(...document.body.children) | |
| .forEach(element => EventHandler.off(element, 'mouseover', noop)) | |
| } | |
| this._activeTrigger[TRIGGER_CLICK] = false | |
| this._activeTrigger[TRIGGER_FOCUS] = false | |
| this._activeTrigger[TRIGGER_HOVER] = false | |
| const isAnimated = this.tip.classList.contains(CLASS_NAME_FADE) | |
| this._queueCallback(complete, this.tip, isAnimated) | |
| this._hoverState = '' | |
| } | |
| update() { | |
| if (this._popper !== null) { | |
| this._popper.update() | |
| } | |
| } | |
| // Protected | |
| isWithContent() { | |
| return Boolean(this.getTitle()) | |
| } | |
| getTipElement() { | |
| if (this.tip) { | |
| return this.tip | |
| } | |
| const element = document.createElement('div') | |
| element.innerHTML = this._config.template | |
| const tip = element.children[0] | |
| this.setContent(tip) | |
| tip.classList.remove(CLASS_NAME_FADE, CLASS_NAME_SHOW) | |
| this.tip = tip | |
| return this.tip | |
| } | |
| setContent(tip) { | |
| this._sanitizeAndSetContent(tip, this.getTitle(), SELECTOR_TOOLTIP_INNER) | |
| } | |
| _sanitizeAndSetContent(template, content, selector) { | |
| const templateElement = SelectorEngine.findOne(selector, template) | |
| if (!content && templateElement) { | |
| templateElement.remove() | |
| return | |
| } | |
| // we use append for html objects to maintain js events | |
| this.setElementContent(templateElement, content) | |
| } | |
| setElementContent(element, content) { | |
| if (element === null) { | |
| return | |
| } | |
| if (isElement(content)) { | |
| content = getElement(content) | |
| // content is a DOM node or a jQuery | |
| if (this._config.html) { | |
| if (content.parentNode !== element) { | |
| element.innerHTML = '' | |
| element.append(content) | |
| } | |
| } else { | |
| element.textContent = content.textContent | |
| } | |
| return | |
| } | |
| if (this._config.html) { | |
| if (this._config.sanitize) { | |
| content = sanitizeHtml(content, this._config.allowList, this._config.sanitizeFn) | |
| } | |
| element.innerHTML = content | |
| } else { | |
| element.textContent = content | |
| } | |
| } | |
| getTitle() { | |
| const title = this._element.getAttribute('data-bs-original-title') || this._config.title | |
| return this._resolvePossibleFunction(title) | |
| } | |
| updateAttachment(attachment) { | |
| if (attachment === 'right') { | |
| return 'end' | |
| } | |
| if (attachment === 'left') { | |
| return 'start' | |
| } | |
| return attachment | |
| } | |
| // Private | |
| _initializeOnDelegatedTarget(event, context) { | |
| return context || this.constructor.getOrCreateInstance(event.delegateTarget, this._getDelegateConfig()) | |
| } | |
| _getOffset() { | |
| const { | |
| offset | |
| } = this._config | |
| if (typeof offset === 'string') { | |
| return offset.split(',').map(val => Number.parseInt(val, 10)) | |
| } | |
| if (typeof offset === 'function') { | |
| return popperData => offset(popperData, this._element) | |
| } | |
| return offset | |
| } | |
| _resolvePossibleFunction(content) { | |
| return typeof content === 'function' ? content.call(this._element) : content | |
| } | |
| _getPopperConfig(attachment) { | |
| const defaultBsPopperConfig = { | |
| placement: attachment, | |
| modifiers: [{ | |
| name: 'flip', | |
| options: { | |
| fallbackPlacements: this._config.fallbackPlacements | |
| } | |
| }, | |
| { | |
| name: 'offset', | |
| options: { | |
| offset: this._getOffset() | |
| } | |
| }, | |
| { | |
| name: 'preventOverflow', | |
| options: { | |
| boundary: this._config.boundary | |
| } | |
| }, | |
| { | |
| name: 'arrow', | |
| options: { | |
| element: `.${this.constructor.NAME}-arrow` | |
| } | |
| }, | |
| { | |
| name: 'onChange', | |
| enabled: true, | |
| phase: 'afterWrite', | |
| fn: data => this._handlePopperPlacementChange(data) | |
| } | |
| ], | |
| onFirstUpdate: data => { | |
| if (data.options.placement !== data.placement) { | |
| this._handlePopperPlacementChange(data) | |
| } | |
| } | |
| } | |
| return { | |
| ...defaultBsPopperConfig, | |
| ...(typeof this._config.popperConfig === 'function' ? this._config.popperConfig(defaultBsPopperConfig) : this._config.popperConfig) | |
| } | |
| } | |
| _addAttachmentClass(attachment) { | |
| this.getTipElement().classList.add(`${this._getBasicClassPrefix()}-${this.updateAttachment(attachment)}`) | |
| } | |
| _getAttachment(placement) { | |
| return AttachmentMap[placement.toUpperCase()] | |
| } | |
| _setListeners() { | |
| const triggers = this._config.trigger.split(' ') | |
| triggers.forEach(trigger => { | |
| if (trigger === 'click') { | |
| EventHandler.on(this._element, this.constructor.Event.CLICK, this._config.selector, event => this.toggle(event)) | |
| } else if (trigger !== TRIGGER_MANUAL) { | |
| const eventIn = trigger === TRIGGER_HOVER ? | |
| this.constructor.Event.MOUSEENTER : | |
| this.constructor.Event.FOCUSIN | |
| const eventOut = trigger === TRIGGER_HOVER ? | |
| this.constructor.Event.MOUSELEAVE : | |
| this.constructor.Event.FOCUSOUT | |
| EventHandler.on(this._element, eventIn, this._config.selector, event => this._enter(event)) | |
| EventHandler.on(this._element, eventOut, this._config.selector, event => this._leave(event)) | |
| } | |
| }) | |
| this._hideModalHandler = () => { | |
| if (this._element) { | |
| this.hide() | |
| } | |
| } | |
| EventHandler.on(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler) | |
| if (this._config.selector) { | |
| this._config = { | |
| ...this._config, | |
| trigger: 'manual', | |
| selector: '' | |
| } | |
| } else { | |
| this._fixTitle() | |
| } | |
| } | |
| _fixTitle() { | |
| const title = this._element.getAttribute('title') | |
| const originalTitleType = typeof this._element.getAttribute('data-bs-original-title') | |
| if (title || originalTitleType !== 'string') { | |
| this._element.setAttribute('data-bs-original-title', title || '') | |
| if (title && !this._element.getAttribute('aria-label') && !this._element.textContent) { | |
| this._element.setAttribute('aria-label', title) | |
| } | |
| this._element.setAttribute('title', '') | |
| } | |
| } | |
| _enter(event, context) { | |
| context = this._initializeOnDelegatedTarget(event, context) | |
| if (event) { | |
| context._activeTrigger[ | |
| event.type === 'focusin' ? TRIGGER_FOCUS : TRIGGER_HOVER | |
| ] = true | |
| } | |
| if (context.getTipElement().classList.contains(CLASS_NAME_SHOW) || context._hoverState === HOVER_STATE_SHOW) { | |
| context._hoverState = HOVER_STATE_SHOW | |
| return | |
| } | |
| clearTimeout(context._timeout) | |
| context._hoverState = HOVER_STATE_SHOW | |
| if (!context._config.delay || !context._config.delay.show) { | |
| context.show() | |
| return | |
| } | |
| context._timeout = setTimeout(() => { | |
| if (context._hoverState === HOVER_STATE_SHOW) { | |
| context.show() | |
| } | |
| }, context._config.delay.show) | |
| } | |
| _leave(event, context) { | |
| context = this._initializeOnDelegatedTarget(event, context) | |
| if (event) { | |
| context._activeTrigger[ | |
| event.type === 'focusout' ? TRIGGER_FOCUS : TRIGGER_HOVER | |
| ] = context._element.contains(event.relatedTarget) | |
| } | |
| if (context._isWithActiveTrigger()) { | |
| return | |
| } | |
| clearTimeout(context._timeout) | |
| context._hoverState = HOVER_STATE_OUT | |
| if (!context._config.delay || !context._config.delay.hide) { | |
| context.hide() | |
| return | |
| } | |
| context._timeout = setTimeout(() => { | |
| if (context._hoverState === HOVER_STATE_OUT) { | |
| context.hide() | |
| } | |
| }, context._config.delay.hide) | |
| } | |
| _isWithActiveTrigger() { | |
| for (const trigger in this._activeTrigger) { | |
| if (this._activeTrigger[trigger]) { | |
| return true | |
| } | |
| } | |
| return false | |
| } | |
| _getConfig(config) { | |
| const dataAttributes = Manipulator.getDataAttributes(this._element) | |
| Object.keys(dataAttributes).forEach(dataAttr => { | |
| if (DISALLOWED_ATTRIBUTES.has(dataAttr)) { | |
| delete dataAttributes[dataAttr] | |
| } | |
| }) | |
| config = { | |
| ...this.constructor.Default, | |
| ...dataAttributes, | |
| ...(typeof config === 'object' && config ? config : {}) | |
| } | |
| config.container = config.container === false ? document.body : getElement(config.container) | |
| if (typeof config.delay === 'number') { | |
| config.delay = { | |
| show: config.delay, | |
| hide: config.delay | |
| } | |
| } | |
| if (typeof config.title === 'number') { | |
| config.title = config.title.toString() | |
| } | |
| if (typeof config.content === 'number') { | |
| config.content = config.content.toString() | |
| } | |
| typeCheckConfig(NAME, config, this.constructor.DefaultType) | |
| if (config.sanitize) { | |
| config.template = sanitizeHtml(config.template, config.allowList, config.sanitizeFn) | |
| } | |
| return config | |
| } | |
| _getDelegateConfig() { | |
| const config = {} | |
| for (const key in this._config) { | |
| if (this.constructor.Default[key] !== this._config[key]) { | |
| config[key] = this._config[key] | |
| } | |
| } | |
| // In the future can be replaced with: | |
| // const keysWithDifferentValues = Object.entries(this._config).filter(entry => this.constructor.Default[entry[0]] !== this._config[entry[0]]) | |
| // `Object.fromEntries(keysWithDifferentValues)` | |
| return config | |
| } | |
| _cleanTipClass() { | |
| const tip = this.getTipElement() | |
| const basicClassPrefixRegex = new RegExp(`(^|\\s)${this._getBasicClassPrefix()}\\S+`, 'g') | |
| const tabClass = tip.getAttribute('class').match(basicClassPrefixRegex) | |
| if (tabClass !== null && tabClass.length > 0) { | |
| tabClass.map(token => token.trim()) | |
| .forEach(tClass => tip.classList.remove(tClass)) | |
| } | |
| } | |
| _getBasicClassPrefix() { | |
| return CLASS_PREFIX | |
| } | |
| _handlePopperPlacementChange(popperData) { | |
| const { | |
| state | |
| } = popperData | |
| if (!state) { | |
| return | |
| } | |
| this.tip = state.elements.popper | |
| this._cleanTipClass() | |
| this._addAttachmentClass(this._getAttachment(state.placement)) | |
| } | |
| _disposePopper() { | |
| if (this._popper) { | |
| this._popper.destroy() | |
| this._popper = null | |
| } | |
| } | |
| // Static | |
| static jQueryInterface(config) { | |
| return this.each(function() { | |
| const data = Tooltip.getOrCreateInstance(this, config) | |
| if (typeof config === 'string') { | |
| if (typeof data[config] === 'undefined') { | |
| throw new TypeError(`No method named "${config}"`) | |
| } | |
| data[config]() | |
| } | |
| }) | |
| } | |
| } | |
| /** | |
| * ------------------------------------------------------------------------ | |
| * jQuery | |
| * ------------------------------------------------------------------------ | |
| * add .Tooltip to jQuery only if jQuery is present | |
| */ | |
| defineJQueryPlugin(Tooltip) | |
| export default Tooltip |