mirror of
https://github.com/element-plus/element-plus.git
synced 2025-01-12 10:45:10 +08:00
74 lines
1.7 KiB
TypeScript
74 lines
1.7 KiB
TypeScript
|
import isServer from '@element-plus/utils/isServer'
|
||
|
import { on } from '@element-plus/utils/dom'
|
||
|
|
||
|
import type { DirectiveBinding, ObjectDirective, VNode, ComponentPublicInstance } from 'vue'
|
||
|
|
||
|
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 => (startClick = e))
|
||
|
on(document, 'mouseup', e => {
|
||
|
for (const { documentHandler } of nodeList.values()) {
|
||
|
documentHandler(e, startClick)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
|
||
|
function createDocumentHandler(
|
||
|
el: HTMLElement,
|
||
|
binding: DirectiveBinding,
|
||
|
): DocumentHandler {
|
||
|
return function(mouseup, mousedown) {
|
||
|
const popperRef = (binding.instance as ComponentPublicInstance<{
|
||
|
popperRef: Nullable<HTMLElement>
|
||
|
}>).popperRef
|
||
|
if (
|
||
|
!binding ||
|
||
|
!binding.instance ||
|
||
|
!mouseup.target ||
|
||
|
!mousedown.target ||
|
||
|
el.contains(mouseup.target as Node) ||
|
||
|
el.contains(mousedown.target as Node) ||
|
||
|
el === mouseup.target ||
|
||
|
(popperRef &&
|
||
|
(popperRef.contains(mouseup.target as Node) ||
|
||
|
popperRef.contains(mousedown.target as Node)))
|
||
|
) {
|
||
|
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
|