clickOutside.ts
2.42 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
import type { ComponentPublicInstance, DirectiveBinding, ObjectDirective } from 'vue'
import { isServer } from '@wry-smile/utils-is'
import { on } from '@/utils/domUtils'
type DocumentHandler = <T extends Event>(mouseup: T, mousedown: T) => void
type FlushList = Map<
HTMLElement,
{
documentHandler: DocumentHandler
bindingFn: (...args: unknown[]) => unknown
}
>
const nodeList: FlushList = new Map()
let startClick: Event
if (!isServer) {
on(document, 'mousedown', (e: Event) => (startClick = e))
on(document, 'mouseup', (e: Event) => {
for (const { documentHandler } of nodeList.values())
documentHandler(e, 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