element-plus/packages/utils/popup-manager.ts
三咲智子 4335b01d6c
refactor(utils)!: refactor global config (#4793)
* refactor(utils)!: remove global config

* refactor(utils): named export PopupManager

* fix tests
2021-12-30 19:31:35 +08:00

228 lines
5.5 KiB
TypeScript

import { getCurrentInstance } from 'vue'
import { isClient } from '@vueuse/core'
import { useGlobalConfig } from '@element-plus/hooks'
import { addClass, removeClass, on } from './dom'
import { EVENT_CODE } from './aria'
import type { Ref } from 'vue'
interface Instance {
closeOnClickModal: Ref<boolean>
closeOnPressEscape: Ref<boolean>
close: () => void
handleClose?: () => void
handleAction?: (action: string) => void
}
type StackFrame = { id: string; zIndex: number; modalClass: string }
interface PopupManager {
getInstance: (id: string) => Instance
zIndex: number
globalInitialZIndex: number
getInitialZIndex: () => number
modalDom?: HTMLElement
modalFade: boolean
modalStack: StackFrame[]
nextZIndex: () => number
register: (id: string, instance: Instance) => void
deregister: (id: string) => void
doOnModalClick: () => void
openModal: (
id: string,
zIndex: number,
dom: HTMLElement,
modalClass: string,
modalFade: boolean
) => void
closeModal: (id: string) => void
}
const onTouchMove = (e: Event) => {
e.preventDefault()
e.stopPropagation()
}
const onModalClick = () => {
PopupManager?.doOnModalClick()
}
let hasModal = false
const getModal = function (): HTMLElement {
if (!isClient) return undefined as any
let modalDom = PopupManager.modalDom
if (modalDom) {
hasModal = true
} else {
hasModal = false
modalDom = document.createElement('div')
PopupManager.modalDom = modalDom
on(modalDom, 'touchmove', onTouchMove)
on(modalDom, 'click', onModalClick)
}
return modalDom
}
const instances = {}
export const PopupManager: PopupManager = {
modalFade: true,
modalDom: undefined,
globalInitialZIndex: 2000,
zIndex: 0,
getInitialZIndex() {
if (!getCurrentInstance()) return this.globalInitialZIndex
return useGlobalConfig('zIndex').value ?? this.globalInitialZIndex
},
getInstance(id) {
return instances[id]
},
register(id, instance) {
if (id && instance) {
instances[id] = instance
}
},
deregister(id) {
if (id) {
instances[id] = null
delete instances[id]
}
},
nextZIndex() {
return this.getInitialZIndex() + ++this.zIndex
},
modalStack: [],
doOnModalClick() {
const topItem = PopupManager.modalStack[PopupManager.modalStack.length - 1]
if (!topItem) return
const instance = PopupManager.getInstance(topItem.id)
if (instance && instance.closeOnClickModal.value) {
instance.close()
}
},
openModal(id, zIndex, dom, modalClass, modalFade) {
if (!isClient) return
if (!id || zIndex === undefined) return
this.modalFade = modalFade
const modalStack = this.modalStack
for (let i = 0, j = modalStack.length; i < j; i++) {
const item = modalStack[i]
if (item.id === id) {
return
}
}
const modalDom = getModal()
addClass(modalDom, 'v-modal')
if (this.modalFade && !hasModal) {
addClass(modalDom, 'v-modal-enter')
}
if (modalClass) {
const classArr = modalClass.trim().split(/\s+/)
classArr.forEach((item) => addClass(modalDom, item))
}
setTimeout(() => {
removeClass(modalDom, 'v-modal-enter')
}, 200)
if (dom && dom.parentNode && dom.parentNode.nodeType !== 11) {
dom.parentNode.appendChild(modalDom)
} else {
document.body.appendChild(modalDom)
}
if (zIndex) {
modalDom.style.zIndex = String(zIndex)
}
modalDom.tabIndex = 0
modalDom.style.display = ''
this.modalStack.push({ id, zIndex, modalClass })
},
closeModal(id) {
const modalStack = this.modalStack
const modalDom = getModal()
if (modalStack.length > 0) {
const topItem = modalStack[modalStack.length - 1]
if (topItem.id === id) {
if (topItem.modalClass) {
const classArr = topItem.modalClass.trim().split(/\s+/)
classArr.forEach((item) => removeClass(modalDom, item))
}
modalStack.pop()
if (modalStack.length > 0) {
modalDom.style.zIndex = `${modalStack[modalStack.length - 1].zIndex}`
}
} else {
for (let i = modalStack.length - 1; i >= 0; i--) {
if (modalStack[i].id === id) {
modalStack.splice(i, 1)
break
}
}
}
}
if (modalStack.length === 0) {
if (this.modalFade) {
addClass(modalDom, 'v-modal-leave')
}
setTimeout(() => {
if (modalStack.length === 0) {
if (modalDom.parentNode) modalDom.parentNode.removeChild(modalDom)
modalDom.style.display = 'none'
// off(modalDom, 'touchmove', onTouchMove)
// off(modalDom, 'click', onModalClick)
PopupManager.modalDom = undefined
}
removeClass(modalDom, 'v-modal-leave')
}, 200)
}
},
}
const getTopPopup = function () {
if (!isClient) return
if (PopupManager.modalStack.length > 0) {
const topPopup = PopupManager.modalStack[PopupManager.modalStack.length - 1]
if (!topPopup) return
const instance = PopupManager.getInstance(topPopup.id)
return instance
}
}
if (isClient) {
// handle `esc` key when the popup is shown
window.addEventListener('keydown', function (event: KeyboardEvent) {
if (event.code === EVENT_CODE.esc) {
const topPopup = getTopPopup()
if (topPopup && topPopup.closeOnPressEscape.value) {
topPopup.handleClose
? topPopup.handleClose()
: topPopup.handleAction
? topPopup.handleAction('cancel')
: topPopup.close()
}
}
})
}