2024-10-13 17:37:01 +00:00
|
|
|
import type { Placement } from '@floating-ui/dom'
|
|
|
|
import { arrow, autoUpdate, offset } from '@floating-ui/dom'
|
|
|
|
import type { Directive, DirectiveBinding } from 'vue'
|
2022-11-13 11:57:39 +00:00
|
|
|
import { updateFloatingUi } from '@/utils'
|
2022-10-25 17:29:56 +00:00
|
|
|
|
|
|
|
type ElementWithTooltip = HTMLElement & {
|
2024-10-13 17:37:01 +00:00
|
|
|
$tooltip?: HTMLDivElement
|
2022-10-25 17:29:56 +00:00
|
|
|
$cleanup?: Closure
|
|
|
|
}
|
|
|
|
|
|
|
|
const getOrCreateTooltip = (el: ElementWithTooltip): HTMLElement => {
|
2024-10-13 17:37:01 +00:00
|
|
|
if (el.$tooltip) {
|
|
|
|
return el.$tooltip
|
|
|
|
}
|
2022-10-25 17:29:56 +00:00
|
|
|
|
|
|
|
el.$tooltip = document.createElement('div')
|
|
|
|
el.$tooltip.classList.add('tooltip')
|
|
|
|
|
|
|
|
const arrow = document.createElement('div')
|
|
|
|
arrow.classList.add('tooltip-arrow')
|
|
|
|
|
|
|
|
const content = document.createElement('div')
|
|
|
|
content.classList.add('tooltip-content')
|
|
|
|
|
|
|
|
el.$tooltip.appendChild(content)
|
|
|
|
el.$tooltip.appendChild(arrow)
|
|
|
|
|
|
|
|
document.body.appendChild(el.$tooltip)
|
|
|
|
|
|
|
|
return el.$tooltip
|
|
|
|
}
|
|
|
|
|
|
|
|
const init = (el: ElementWithTooltip, binding: DirectiveBinding) => {
|
|
|
|
const $tooltip = getOrCreateTooltip(el)
|
|
|
|
|
|
|
|
// make sure the actual title is removed from the element, but keep a backup for the updated() hook calls
|
2024-10-13 17:37:01 +00:00
|
|
|
$tooltip.querySelector<HTMLDivElement>('.tooltip-content')!.textContent = binding.value
|
|
|
|
|| el.title
|
|
|
|
|| el.getAttribute('data-title')
|
|
|
|
|| el.textContent
|
2022-10-25 17:29:56 +00:00
|
|
|
|
|
|
|
if (el.title && !el.getAttribute('data-title')) {
|
|
|
|
el.setAttribute('data-title', el.title)
|
|
|
|
el.removeAttribute('title')
|
|
|
|
}
|
|
|
|
|
|
|
|
const $arrow = $tooltip.querySelector<HTMLDivElement>('.tooltip-arrow')!
|
|
|
|
|
|
|
|
let placement: Placement = 'bottom'
|
|
|
|
|
|
|
|
;(['left', 'right', 'top', 'bottom'] as Placement[]).forEach(p => {
|
|
|
|
if (binding.modifiers[p]) {
|
|
|
|
placement = p
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-11-13 11:57:39 +00:00
|
|
|
const update = async () => await updateFloatingUi(el, $tooltip, {
|
|
|
|
placement,
|
|
|
|
middleware: [
|
|
|
|
arrow({ element: $arrow }),
|
2024-10-13 17:37:01 +00:00
|
|
|
offset(8),
|
|
|
|
],
|
2022-11-13 11:57:39 +00:00
|
|
|
}, $arrow)
|
2022-10-25 17:29:56 +00:00
|
|
|
|
|
|
|
el.$cleanup = el.$cleanup || autoUpdate(el, $tooltip, update)
|
|
|
|
|
|
|
|
const showTooltip = async () => {
|
|
|
|
$tooltip.classList.add('show')
|
|
|
|
await update()
|
|
|
|
}
|
|
|
|
|
|
|
|
const hideTooltip = () => $tooltip.classList.remove('show')
|
|
|
|
|
|
|
|
el.addEventListener('mouseenter', showTooltip)
|
|
|
|
el.addEventListener('focus', showTooltip)
|
|
|
|
el.addEventListener('mouseleave', hideTooltip)
|
|
|
|
el.addEventListener('blur', hideTooltip)
|
|
|
|
}
|
|
|
|
|
|
|
|
export const tooltip: Directive = {
|
|
|
|
mounted: init,
|
|
|
|
updated: init,
|
|
|
|
|
2024-04-04 22:20:42 +00:00
|
|
|
beforeUnmount: (el: ElementWithTooltip) => {
|
2022-10-25 17:29:56 +00:00
|
|
|
el.$cleanup && el.$cleanup()
|
2022-10-27 13:33:32 +00:00
|
|
|
el.$tooltip && document.body.removeChild(el.$tooltip)
|
2024-10-13 17:37:01 +00:00
|
|
|
},
|
2022-10-25 17:29:56 +00:00
|
|
|
}
|