参考答案:
组件就是把图形、非图形的各种逻辑均抽象为一个统一的概念(组件)来实现开发的模式
现在有一个场景,点击新增与编辑都弹框出来进行填写,功能上大同小异,可能只是标题内容或者是显示的主体内容稍微不同
这时候就没必要写两个组件,只需要根据传入的参数不同,组件显示不同内容即可
这样,下次开发相同界面程序时就可以写更少的代码,意义着更高的开发效率,更少的 Bug
和更少的程序体积
实现一个Modal
组件,首先确定需要完成的内容:
遮罩层
标题内容
主体内容
确定和取消按钮
主体内容需要灵活,所以可以是字符串,也可以是一段 html
代码
特点是它们在当前vue
实例之外独立存在,通常挂载于body
之上
除了通过引入import
的形式,我们还可通过API
的形式进行组件的调用
还可以包括配置全局样式、国际化、与typeScript
结合
首先看看大致流程:
目录结构
组件内容
实现 API 形式
事件处理
其他完善
Modal
组件相关的目录结构
├── plugins
│ └── modal
│ ├── Content.tsx // 维护 Modal 的内容,用于 h 函数和 jsx 语法
│ ├── Modal.vue // 基础组件
│ ├── config.ts // 全局默认配置
│ ├── index.ts // 入口
│ ├── locale // 国际化相关
│ │ ├── index.ts
│ │ └── lang
│ │ ├── en-US.ts
│ │ ├── zh-CN.ts
│ │ └── zh-TW.ts
│ └── modal.type.ts // ts类型声明相关
因为 Modal 会被 app.use(Modal)
调用作为一个插件,所以都放在plugins
目录下
首先实现modal.vue
的主体显示内容大致如下
1<Teleport to="body" :disabled="!isTeleport"> 2 <div v-if="modelValue" class="modal"> 3 <div 4 class="mask" 5 :style="style" 6 @click="maskClose && !loading && handleCancel()" 7 ></div> 8 <div class="modal__main"> 9 <div class="modal__title line line--b"> 10 <span>{{ title || t("r.title") }}</span> 11 <span 12 v-if="close" 13 :title="t('r.close')" 14 class="close" 15 @click="!loading && handleCancel()" 16 >✕</span 17 > 18 </div> 19 <div class="modal__content"> 20 <Content v-if="typeof content === 'function'" :render="content" /> 21 <slot v-else> 22 {{ content }} 23 </slot> 24 </div> 25 <div class="modal__btns line line--t"> 26 <button :disabled="loading" @click="handleConfirm"> 27 <span class="loading" v-if="loading"> ❍ </span>{{ t("r.confirm") }} 28 </button> 29 <button @click="!loading && handleCancel()"> 30 {{ t("r.cancel") }} 31 </button> 32 </div> 33 </div> 34 </div> 35</Teleport>
最外层上通过Vue3 Teleport
内置组件进行包裹,其相当于传送门,将里面的内容传送至body
之上
并且从DOM
结构上来看,把modal
该有的内容(遮罩层、标题、内容、底部按钮)都实现了
关于主体内容
1<div class="modal__content"> 2 <Content v-if="typeof content==='function'" 3 :render="content" /> 4 <slot v-else> 5 {{content}} 6 </slot> 7</div>
可以看到根据传入content
的类型不同,对应显示不同得到内容
最常见的则是通过调用字符串和默认插槽的形式
1// 默认插槽 2<Modal v-model="show" 3 title="演示 slot"> 4 <div>hello world~</div> 5</Modal> 6 7// 字符串 8<Modal v-model="show" 9 title="演示 content" 10 content="hello world~" />
通过 API 形式调用Modal
组件的时候,content
可以使用下面两种
1$modal.show({ 2 title: '演示 h 函数', 3 content(h) { 4 return h( 5 'div', 6 { 7 style: 'color:red;', 8 onClick: ($event: Event) => console.log('clicked', $event.target) 9 }, 10 'hello world ~' 11 ); 12 } 13});
1$modal.show({ 2 title: '演示 jsx 语法', 3 content() { 4 return ( 5 <div 6 onClick={($event: Event) => console.log('clicked', $event.target)} 7 > 8 hello world ~ 9 </div> 10 ); 11 } 12});
那么组件如何实现API
形式调用Modal
组件呢?
在Vue2
中,我们可以借助Vue
实例以及Vue.extend
的方式获得组件实例,然后挂载到body
上
1import Modal from './Modal.vue'; 2const ComponentClass = Vue.extend(Modal); 3const instance = new ComponentClass({ el: document.createElement("div") }); 4document.body.appendChild(instance.$el);
虽然Vue3
移除了Vue.extend
方法,但可以通过createVNode
实现
1import Modal from './Modal.vue'; 2const container = document.createElement('div'); 3const vnode = createVNode(Modal); 4render(vnode, container); 5const instance = vnode.component; 6document.body.appendChild(container);
在Vue2
中,可以通过this
的形式调用全局 API
1export default { 2 install(vue) { 3 vue.prototype.$create = create 4 } 5}
而在 Vue3 的 setup
中已经没有 this
概念了,需要调用app.config.globalProperties
挂载到全局
1export default { 2 install(app) { 3 app.config.globalProperties.$create = create 4 } 5}
下面再看看看Modal
组件内部是如何处理「确定」「取消」事件的,既然是Vue3
,当然采用Compositon API
形式
1// Modal.vue 2setup(props, ctx) { 3 let instance = getCurrentInstance(); // 获得当前组件实例 4 onBeforeMount(() => { 5 instance._hub = { 6 'on-cancel': () => {}, 7 'on-confirm': () => {} 8 }; 9 }); 10 11 const handleConfirm = () => { 12 ctx.emit('on-confirm'); 13 instance._hub['on-confirm'](); 14 }; 15 const handleCancel = () => { 16 ctx.emit('on-cancel'); 17 ctx.emit('update:modelValue', false); 18 instance._hub['on-cancel'](); 19 }; 20 21 return { 22 handleConfirm, 23 handleCancel 24 }; 25}
在上面代码中,可以看得到除了使用传统emit
的形式使父组件监听,还可通过_hub
属性中添加 on-cancel
,on-confirm
方法实现在API
中进行监听
1app.config.globalProperties.$modal = { 2 show({}) { 3 /* 监听 确定、取消 事件 */ 4 } 5}
下面再来目睹下_hub
是如何实现
1// index.ts 2app.config.globalProperties.$modal = { 3 show({ 4 /* 其他选项 */ 5 onConfirm, 6 onCancel 7 }) { 8 /* ... */ 9 10 const { props, _hub } = instance; 11 12 const _closeModal = () => { 13 props.modelValue = false; 14 container.parentNode!.removeChild(container); 15 }; 16 // 往 _hub 新增事件的具体实现 17 Object.assign(_hub, { 18 async 'on-confirm'() { 19 if (onConfirm) { 20 const fn = onConfirm(); 21 // 当方法返回为 Promise 22 if (fn && fn.then) { 23 try { 24 props.loading = true; 25 await fn; 26 props.loading = false; 27 _closeModal(); 28 } catch (err) { 29 // 发生错误时,不关闭弹框 30 console.error(err); 31 props.loading = false; 32 } 33 } else { 34 _closeModal(); 35 } 36 } else { 37 _closeModal(); 38 } 39 }, 40 'on-cancel'() { 41 onCancel && onCancel(); 42 _closeModal(); 43 } 44 }); 45} 46};
关于组件实现国际化、与typsScript
结合,大家可以根据自身情况在此基础上进行更改
最近更新时间:2024-08-10