在业务中,我们经常会遇到各种各样的弹窗组件,一般我们都是直接将其封装成一个组件,通过v-model
或者v-if
之类的手段进行控制, But 很容易就出现下面这种情况
<DetailModal v-model="detailVisible" />
<ConfirmModal v-model="confirmVisible" />
<BindModal v-model="bindVisible />
现上述这种情况,我们需要同时维护多个弹窗的隐藏显示,然后要写很多个 toggle BalaBala。 于是乎, 我们可以看到很多UI框架的封装Modal都提供了函数调用的形式,直接通过函数就可以控制弹窗。
this.$Modal.confirm({
content: '详情弹窗‘
})
上述的方式,我们可以更加简洁地维护弹窗状态,而不用去同时维护对应的html和状态,简单看下下面的sample
import Vue, { CreateElement } from "vue";
import { Button } from "view-design";
let Instance!: any;
export default function deleteModal(props: any) {
if (!Instance) {
Instance = new Vue({
data() {
return {
visible: false,
onOk: () => {},
loading: false,
title: ""
};
},
methods: {
show(options: any) {
Instance.visible = true;
Instance.onOk = options.onOk;
Instance.loading = options.loading;
Instance.title = options.title;
Instance.content = options.content;
}
},
render(this: any, h: CreateElement) {
const okButton = (scope: any) =>
h(
Button,
{
props: {
type: "error",
ghost: true,
loading: this.loading
},
slot: "ok",
on: {
click: () => {
this.onOk(scope.close);
}
}
},
["删除"]
);
return h(
"Modal",
{
props: {
...props,
content: [],
title: this.title,
value: this.visible
},
on: {
input: (status: boolean) => {
this.visible = status;
}
},
scopedSlots: {
ok: okButton
}
},
[this.content]
);
}
});
const component = Instance.$mount();
document.body.appendChild(component.$el);
const modal = Instance.$children[0];
}
Instance.show(props);
return Instance;
}
上述的Sample的场景是删除提示的按钮, 通过 deleteModal
的方式我们就可以调用起对应的窗口, 而不需要在父组件中定义相关的弹窗组件,
其主要原因是通过Vue.$mount
获取到对应组件的dom元素,直接挂载到docume.body上, 通过控制实例来执行相关的操作, 但以上的代码还有很多需要改进的地方,
待续...