[improvement] popup mixin ts (#3022)

This commit is contained in:
neverland 2019-03-21 10:08:26 +08:00 committed by GitHub
parent e04322a2e6
commit 7eda33eb44
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 100 additions and 91 deletions

View File

@ -1,9 +0,0 @@
export default {
zIndex: 2000,
stack: [],
lockCount: 0,
get top() {
return this.stack[this.stack.length - 1];
}
};

View File

@ -0,0 +1,17 @@
import { OverlayConfig } from './overlay';
export type StackItem = {
vm: any;
target: HTMLElement;
config: OverlayConfig;
};
export const context = {
zIndex: 2000,
lockCount: 0,
stack: [] as StackItem[],
get top(): StackItem {
return this.stack[this.stack.length - 1];
}
};

View File

@ -1,7 +1,7 @@
import manager from './manager';
import context from './context';
import { context } from './context';
import { TouchMixin } from '../touch';
import { on, off } from '../../utils/event';
import { openOverlay, closeOverlay } from './overlay';
import { getScrollEventTarget } from '../../utils/scroll';
export const PopupMixin = {
@ -134,7 +134,7 @@ export const PopupMixin = {
}
this.opened = false;
manager.close(this);
closeOverlay(this);
this.$emit('input', false);
},
@ -187,13 +187,13 @@ export const PopupMixin = {
renderOverlay() {
if (this.overlay) {
manager.open(this, {
openOverlay(this, {
zIndex: context.zIndex++,
className: this.overlayClass,
customStyle: this.overlayStyle
});
} else {
manager.close(this);
closeOverlay(this);
}
this.$nextTick(() => {

View File

@ -1,77 +0,0 @@
import context from './context';
import Overlay from '../../overlay';
import { mount } from '../../utils/functional';
const defaultConfig = {
className: '',
customStyle: {}
};
export default {
open(vm, config) {
/* istanbul ignore next */
if (!context.stack.some(item => item.vm === vm)) {
const el = vm.$el;
const target = el && el.parentNode ? el.parentNode : document.body;
context.stack.push({ vm, config, target });
this.update();
}
},
close(vm) {
const { stack } = context;
if (stack.length) {
if (context.top.vm === vm) {
stack.pop();
this.update();
} else {
context.stack = stack.filter(item => item.vm !== vm);
}
}
},
update() {
let { modal } = context;
if (!modal) {
modal = mount(Overlay, {
on: {
click: this.onClick
}
});
context.modal = modal;
}
if (modal.$el.parentNode) {
modal.visible = false;
}
if (context.top) {
const { target, config } = context.top;
target.appendChild(modal.$el);
Object.assign(modal, defaultConfig, config, {
visible: true
});
}
},
// close popup when click modal && closeOnClickOverlay is true
onClick() {
/* istanbul ignore else */
if (context.top) {
const { vm } = context.top;
vm.$emit('click-overlay');
if (vm.closeOnClickOverlay) {
if (vm.onClickOverlay) {
vm.onClickOverlay();
} else {
vm.close();
}
}
}
}
};

View File

@ -0,0 +1,78 @@
import Overlay from '../../overlay';
import { context } from './context';
import { mount } from '../../utils/functional';
export type OverlayConfig = {
zIndex?: number;
className?: string;
customStyle?: object[] | object;
};
const defaultConfig: OverlayConfig = {
className: '',
customStyle: {}
};
let overlay: any;
// close popup when click overlay && closeOnClickOverlay is true
function onClickOverlay(): void {
if (context.top) {
const { vm } = context.top;
vm.$emit('click-overlay');
if (vm.closeOnClickOverlay) {
if (vm.onClickOverlay) {
vm.onClickOverlay();
} else {
vm.close();
}
}
}
}
function updateOverlay(): void {
if (!overlay) {
overlay = mount(Overlay, {
on: {
click: onClickOverlay
}
});
}
if (overlay.$el.parentNode) {
overlay.visible = false;
}
if (context.top) {
const { target, config } = context.top;
target.appendChild(overlay.$el);
Object.assign(overlay, defaultConfig, config, {
visible: true
});
}
}
export function openOverlay(vm: any, config: OverlayConfig): void {
/* istanbul ignore next */
if (!context.stack.some(item => item.vm === vm)) {
const el = vm.$el;
const target = el && el.parentNode ? el.parentNode : document.body;
context.stack.push({ vm, config, target });
updateOverlay();
}
}
export function closeOverlay(vm: any): void {
const { stack } = context;
if (stack.length) {
if (context.top.vm === vm) {
stack.pop();
updateOverlay();
} else {
context.stack = stack.filter(item => item.vm !== vm);
}
}
}