| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879 |
- import type { ComponentPublicInstance, DirectiveBinding, ObjectDirective } from 'vue'
- import { on } from '@/utils/domUtils'
- import { isServer } from '@/utils/is'
- type DocumentHandler = <T extends MouseEvent>(mouseup: T, mousedown: T) => void
- type FlushList = Map<
- HTMLElement,
- {
- documentHandler: DocumentHandler
- bindingFn: (...args: unknown[]) => unknown
- }
- >
- const nodeList: FlushList = new Map()
- let startClick: MouseEvent
- if (!isServer) {
- on(document, 'mousedown', (e: Event) => (startClick = e as MouseEvent))
- on(document, 'mouseup', (e: Event) => {
- for (const { documentHandler } of nodeList.values())
- documentHandler(e as MouseEvent, startClick)
- })
- }
- function createDocumentHandler(el: HTMLElement, binding: DirectiveBinding): DocumentHandler {
- let excludes: HTMLElement[] = []
- if (Array.isArray(binding.arg)) {
- excludes = binding.arg
- }
- else {
- // due to current implementation on binding type is wrong the type casting is necessary here
- excludes.push(binding.arg as unknown as HTMLElement)
- }
- return function (mouseup, mousedown) {
- const popperRef = (
- binding.instance as ComponentPublicInstance<{
- popperRef: Nullable<HTMLElement>
- }>
- ).popperRef
- const mouseUpTarget = mouseup.target as Node
- const mouseDownTarget = mousedown.target as Node
- const isBound = !binding || !binding.instance
- const isTargetExists = !mouseUpTarget || !mouseDownTarget
- const isContainedByEl = el.contains(mouseUpTarget) || el.contains(mouseDownTarget)
- const isSelf = el === mouseUpTarget
- const isTargetExcluded
- = (excludes.length && excludes.some(item => item?.contains(mouseUpTarget)))
- || (excludes.length && excludes.includes(mouseDownTarget as HTMLElement))
- const isContainedByPopper = popperRef && (popperRef.contains(mouseUpTarget) || popperRef.contains(mouseDownTarget))
- if (isBound || isTargetExists || isContainedByEl || isSelf || isTargetExcluded || isContainedByPopper)
- return
- binding.value()
- }
- }
- const ClickOutside: ObjectDirective = {
- beforeMount(el, binding) {
- nodeList.set(el, {
- documentHandler: createDocumentHandler(el, binding),
- bindingFn: binding.value,
- })
- },
- updated(el, binding) {
- nodeList.set(el, {
- documentHandler: createDocumentHandler(el, binding),
- bindingFn: binding.value,
- })
- },
- unmounted(el) {
- nodeList.delete(el)
- },
- }
- export default ClickOutside
|