mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
135 lines
3.4 KiB
JavaScript
135 lines
3.4 KiB
JavaScript
import Vue from 'vue';
|
|
import PopupContext from './popup-context';
|
|
|
|
const getModal = function() {
|
|
if (Vue.prototype.$isServer) return;
|
|
let modalDom = PopupContext.getContext('modalDom');
|
|
|
|
if (modalDom) {
|
|
PopupContext.setContext('hasModal', true);
|
|
} else {
|
|
PopupContext.setContext('hasModal', false);
|
|
|
|
modalDom = document.createElement('div');
|
|
PopupContext.setContext('modalDom', modalDom);
|
|
|
|
modalDom.addEventListener('touchmove', function(event) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
});
|
|
|
|
modalDom.addEventListener('click', function() {
|
|
PopupManager.handleOverlayClick && PopupManager.handleOverlayClick();
|
|
});
|
|
}
|
|
|
|
return modalDom;
|
|
};
|
|
|
|
const PopupManager = {
|
|
nextZIndex() {
|
|
return PopupContext.plusKeyByOne('zIndex');
|
|
},
|
|
|
|
getInstance(id) {
|
|
return PopupContext.getContext('instances')[id];
|
|
},
|
|
|
|
register(id, instance) {
|
|
if (id && instance) {
|
|
const instances = PopupContext.getContext('instances');
|
|
instances[id] = instance;
|
|
}
|
|
},
|
|
|
|
deregister(id) {
|
|
if (id) {
|
|
const instances = PopupContext.getContext('instances');
|
|
instances[id] = null;
|
|
delete instances[id];
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 遮罩层点击回调,`closeOnClickOverlay`为`true`时会关闭当前`popup`
|
|
*/
|
|
handleOverlayClick() {
|
|
const modalStack = PopupContext.getContext('modalStack');
|
|
const topModal = modalStack[modalStack.length - 1];
|
|
if (!topModal) return;
|
|
|
|
const instance = PopupManager.getInstance(topModal.id);
|
|
if (instance && instance.closeOnClickOverlay) {
|
|
instance.close();
|
|
}
|
|
},
|
|
|
|
openModal(id, zIndex, dom) {
|
|
if (!id || zIndex === undefined) return;
|
|
|
|
const modalStack = PopupContext.getContext('modalStack');
|
|
|
|
for (let i = 0, len = modalStack.length; i < len; i++) {
|
|
const item = modalStack[i];
|
|
if (item.id === id) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
const modalDom = getModal();
|
|
|
|
modalDom.classList.add('van-modal');
|
|
|
|
let domParentNode;
|
|
if (dom && dom.parentNode && dom.parentNode.nodeType !== 11) {
|
|
domParentNode = dom.parentNode;
|
|
} else {
|
|
domParentNode = document.body;
|
|
}
|
|
domParentNode.appendChild(modalDom);
|
|
|
|
if (zIndex) {
|
|
modalDom.style.zIndex = zIndex;
|
|
}
|
|
modalDom.style.display = '';
|
|
|
|
modalStack.push({ id: id, zIndex: zIndex, parentNode: domParentNode });
|
|
},
|
|
|
|
closeModal(id) {
|
|
const modalStack = PopupContext.getContext('modalStack');
|
|
const modalDom = getModal();
|
|
|
|
if (modalStack.length > 0) {
|
|
const topItem = modalStack[modalStack.length - 1];
|
|
if (topItem.id === id) {
|
|
modalStack.pop();
|
|
if (modalStack.length > 0) {
|
|
modalDom.style.zIndex = modalStack[modalStack.length - 1].zIndex;
|
|
modalDom.parentNode.removeChild(modalDom);
|
|
const currModalParent = modalStack[0].parentNode;
|
|
currModalParent && currModalParent.appendChild(modalDom);
|
|
}
|
|
} else {
|
|
for (let i = modalStack.length - 1; i >= 0; i--) {
|
|
if (modalStack[i].id === id) {
|
|
modalStack.splice(i, 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (modalStack.length === 0) {
|
|
setTimeout(() => {
|
|
if (modalDom.parentNode) modalDom.parentNode.removeChild(modalDom);
|
|
|
|
modalDom.style.display = 'none';
|
|
this.modalDom = null;
|
|
}, 200);
|
|
}
|
|
}
|
|
};
|
|
|
|
export default PopupManager;
|