mirror of
https://github.com/element-plus/element-plus.git
synced 2025-01-12 10:45:10 +08:00
55348b30b6
* style: use prettier * style: just prettier format, no code changes * style: eslint fix object-shorthand, prefer-const * style: fix no-void * style: no-console
235 lines
5.4 KiB
TypeScript
235 lines
5.4 KiB
TypeScript
import isServer from './isServer'
|
|
import * as configs from './config'
|
|
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 IPopupManager {
|
|
getInstance: (id: string) => Instance
|
|
zIndex: 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
|
|
let zIndex: number
|
|
|
|
const getModal = function (): HTMLElement {
|
|
if (isServer) return
|
|
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 = {}
|
|
|
|
const PopupManager: IPopupManager = {
|
|
modalFade: true,
|
|
modalDom: undefined,
|
|
zIndex,
|
|
|
|
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 ++PopupManager.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 (isServer) 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)
|
|
}
|
|
},
|
|
}
|
|
|
|
Object.defineProperty(PopupManager, 'zIndex', {
|
|
configurable: true,
|
|
get() {
|
|
if (zIndex === undefined) {
|
|
zIndex = (configs.getConfig('zIndex') as number) || 2000
|
|
}
|
|
return zIndex
|
|
},
|
|
set(value) {
|
|
zIndex = value
|
|
},
|
|
})
|
|
|
|
const getTopPopup = function () {
|
|
if (isServer) 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 (!isServer) {
|
|
// handle `esc` key when the popup is shown
|
|
on(window, '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()
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
export default PopupManager
|