2024-10-13 17:37:01 +00:00
|
|
|
import type { Ref } from 'vue'
|
|
|
|
import { isRef } from 'vue'
|
|
|
|
import type { Placement } from '@floating-ui/dom'
|
|
|
|
import { arrow as arrowMiddleware, autoUpdate, flip, offset } from '@floating-ui/dom'
|
2022-11-13 11:57:39 +00:00
|
|
|
import { updateFloatingUi } from '@/utils'
|
2022-11-12 21:38:31 +00:00
|
|
|
|
2024-10-13 17:37:01 +00:00
|
|
|
export interface Config {
|
|
|
|
placement: Placement
|
|
|
|
useArrow: boolean
|
|
|
|
autoTrigger: boolean
|
2022-11-12 21:38:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export const useFloatingUi = (
|
2022-12-02 16:17:37 +00:00
|
|
|
reference: HTMLElement | Ref<HTMLElement | undefined>,
|
|
|
|
floating: HTMLElement | Ref<HTMLElement | undefined>,
|
2024-10-13 17:37:01 +00:00
|
|
|
config: Partial<Config> = {},
|
2022-11-12 21:38:31 +00:00
|
|
|
) => {
|
2024-04-23 21:01:27 +00:00
|
|
|
const extractRef = <T extends HTMLElement | Ref<HTMLElement | undefined>> (ref: T): HTMLElement => {
|
2022-12-02 16:17:37 +00:00
|
|
|
if (isRef(ref) && !ref.value) {
|
|
|
|
throw new TypeError('Reference element is not defined')
|
|
|
|
}
|
|
|
|
|
|
|
|
return isRef(ref) ? ref.value! : ref
|
|
|
|
}
|
|
|
|
|
2022-11-12 21:38:31 +00:00
|
|
|
const mergedConfig: Config = Object.assign({
|
|
|
|
placement: 'bottom',
|
|
|
|
useArrow: true,
|
2024-10-13 17:37:01 +00:00
|
|
|
autoTrigger: true,
|
2022-11-12 21:38:31 +00:00
|
|
|
}, config)
|
|
|
|
|
|
|
|
let _cleanUp: Closure
|
|
|
|
let _show: Closure
|
|
|
|
let _hide: Closure
|
|
|
|
let _trigger: Closure
|
|
|
|
|
|
|
|
const setup = () => {
|
2022-12-02 16:17:37 +00:00
|
|
|
const referenceElement = extractRef(reference)
|
|
|
|
const floatingElement = extractRef(floating)
|
2022-11-12 21:38:31 +00:00
|
|
|
|
2022-12-02 16:17:37 +00:00
|
|
|
floatingElement.style.display = 'none'
|
2022-11-12 21:38:31 +00:00
|
|
|
|
2022-11-13 11:57:39 +00:00
|
|
|
const middleware = [
|
|
|
|
flip(),
|
2024-10-13 17:37:01 +00:00
|
|
|
offset(6),
|
2022-11-13 11:57:39 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
let arrow: HTMLElement
|
2022-11-12 21:38:31 +00:00
|
|
|
|
|
|
|
if (mergedConfig.useArrow) {
|
|
|
|
arrow = document.createElement('div')
|
|
|
|
arrow.className = 'arrow'
|
2022-12-02 16:17:37 +00:00
|
|
|
floatingElement.appendChild(arrow)
|
2022-11-12 21:38:31 +00:00
|
|
|
|
|
|
|
middleware.push(arrowMiddleware({
|
|
|
|
element: arrow,
|
2024-10-13 17:37:01 +00:00
|
|
|
padding: 6,
|
2022-11-12 21:38:31 +00:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2022-12-02 16:17:37 +00:00
|
|
|
const update = async () => await updateFloatingUi(referenceElement, floatingElement, {
|
2022-11-13 11:57:39 +00:00
|
|
|
placement: mergedConfig.placement,
|
2024-10-13 17:37:01 +00:00
|
|
|
middleware,
|
2022-11-13 11:57:39 +00:00
|
|
|
}, arrow)
|
2022-11-12 21:38:31 +00:00
|
|
|
|
2022-12-02 16:17:37 +00:00
|
|
|
_cleanUp = autoUpdate(referenceElement, floatingElement, update)
|
2022-11-12 21:38:31 +00:00
|
|
|
|
|
|
|
_show = async () => {
|
2022-12-02 16:17:37 +00:00
|
|
|
floatingElement.style.display = 'block'
|
2022-11-12 21:38:31 +00:00
|
|
|
await update()
|
|
|
|
}
|
|
|
|
|
2022-12-02 16:17:37 +00:00
|
|
|
_hide = () => (floatingElement.style.display = 'none')
|
|
|
|
_trigger = () => floatingElement.style.display === 'none' ? _show() : _hide()
|
2022-11-12 21:38:31 +00:00
|
|
|
|
|
|
|
if (mergedConfig.autoTrigger) {
|
2022-12-02 16:17:37 +00:00
|
|
|
referenceElement.addEventListener('mouseenter', _show)
|
|
|
|
referenceElement.addEventListener('focus', _show)
|
|
|
|
referenceElement.addEventListener('mouseleave', _hide)
|
|
|
|
referenceElement.addEventListener('blur', _hide)
|
2022-11-12 21:38:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
setup,
|
|
|
|
teardown: () => _cleanUp(),
|
|
|
|
show: () => _show(),
|
|
|
|
hide: () => _hide(),
|
2024-10-13 17:37:01 +00:00
|
|
|
trigger: () => _trigger(),
|
2022-11-12 21:38:31 +00:00
|
|
|
}
|
|
|
|
}
|