用 Vue3.0 写过组件吗?如果想实现一个 Modal 你会 怎么设计?

一、组件设计

组件就是把图形、非图形的各种逻辑均抽象为一个统一的概念(组件)来实现开发的模式
现在有一个场景,点击新增与编辑都弹框出来进行填写,功能上大同小异,可能只是标题内容或者是显示的主体内容稍微不同
这时候就没必要写两个组件,只需要根据传入的参数不同,组件显示不同内容即可这样,下次开发相同界面程序时就可以写更少的代码,意义着更高的开发效率,更少的 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 的主体显示内容大致如下

<Teleport to="body" :disabled="!isTeleport">
   <div v-if="modelValue" class="modal">
 <div
  class="mask"
 :style="style"
 @click="maskClose && !loading && handleCancel()"
 ></div>
 <div class="modal__main">
 <div class="modal__title line line--b">
 <span>{{ title || t("r.title") }}</span>
  <span
   v-if="close"
   :title="t('r.close')"
   class="close"
 @click="!loading && handleCancel()"
  >✕</span
  >
 </div>
 <div class="modal__content">
 <Content v-if="typeof content === 'function'" :render="
 content" />
  <slot v-else>
   {{ content }}
  </slot>
  </div>
  <div class="modal__btns line line--t">
  <button :disabled="loading" @click="handleConfirm">
   <span class="loading" v-if="loading"> ❍ </span>{{ 
   t("r.confirm") }}
  </button>
    <button @click="!loading && handleCancel()">
   {{ t("r.cancel") }}
   </button>
   </div>
   </div>
   </div>
 </Teleport>

最外层上通过 Vue3 Teleport 内置组件进行包裹,其相当于传送门,将里面的内容传送至 body 之上
并且从 DOM 结构上来看,把 modal 该有的内容(遮罩层、标题、内容、底部按钮)
都实现了关于主体内容

<div class="modal__content">
  <Content v-if="typeof content==='function'"
 :render="content" />
  <slot v-else>
   {{content}}
  </slot>
</div>

可以看到根据传入 content 的类型不同,对应显示不同得到内容
最常见的则是通过调用字符串和默认插槽的形式

// 默认插槽
<Modal v-model="show"
  title="演示 slot">
   <div>hello world~</div>
</Modal>

 // 字符串
 <Modal v-model="show"
  title="演示 content"
  content="hello world~" />

通过 API 形式调用 Modal 组件的时候,content 可以使用下面两种

  • h 函数

     $modal.show({
      title: '演示 h 函数',
      content(h) {
      return h(
        'div',
        style: 'color:red;',
         onClick: ($event: Event) => console.log('clicked', $event.target)
        },
       'hello world ~'
          );
       }
      });
    
    • JSX

        $modal.show({
        title: '演示 jsx 语法',
        content() {
        return (
         <div
            onClick={($event: Event) => console.log('clicked', $event.target)} >
            hello world ~
        </div>
      
实现 API 形式

那么组件如何实现 API 形式调用 Modal 组件呢?

在 Vue2 中,我们可以借助 Vue 实例以及 Vue.extend 的方式获得组件实例,然后
挂载到 body 上

import Modal from './Modal.vue';
const ComponentClass = Vue.extend(Modal);
const instance = new ComponentClass({ el: document.createElement("div")
 });
document.body.appendChild(instance.$el);

虽然 Vue3 移除了 Vue.extend 方法,但可以通过 createVNode 实现

import Modal from './Modal.vue';
const container = document.createElement('div');
const vnode = createVNode(Modal);
render(vnode, container);
const instance = vnode.component;
document.body.appendChild(container);

在Vue2中,可以通过 this 的形式调用全局API

 export default {
  install(vue) {
     vue.prototype.$create = create
    }
  }

而在 Vue3 的 setup 中已经没有 this 概念了,需要调用

app.config.globalProperties 挂载到全局

export default {
   install(app) {
   app.config.globalProperties.$create = create
  }
}
事件处理

下面再看看看 Modal 组件内部是如何处理「确定」「取消」事件的,既然是 Vue3 ,当然采用 Compositon API 形式

 // Modal.vue
 setup(props, ctx) {
   let instance = getCurrentInstance(); // 获得当前组件实例
   onBeforeMount(() => {
   instance._hub = {
   'on-cancel': () => {},
   'on-confirm': () => {}
    };
 });

  const handleConfirm = () => {
     ctx.emit('on-confirm');
     instance._hub['on-confirm']();
 };
 const handleCancel = () => {
  ctx.emit('on-cancel');
  ctx.emit('update:modelValue', false);
  instance._hub['on-cancel']();
 };
 return {
  handleConfirm,
  handleCancel
 };
}

在上面代码中,可以看得到除了使用传统 emit 的形式使父组件监听,还可通过
_hub 属性中添加 on-cancel , on-confirm 方法实现在 API 中进行监听

 app.config.globalProperties.$modal = {
    show({}) {
   /* 监听 确定、取消 事件 */
   }
 }

下面再来目睹下 _hub 是如何实现

// index.ts
  app.config.globalProperties.$modal = {
   show({
   /* 其他选项 */
   onConfirm,
    onCancel
   }) {
   /* ... */
  const { props, _hub } = instance;
  const _closeModal = () => {
  props.modelValue = false;
  container.parentNode!.removeChild(container);
 };
 // 往 _hub 新增事件的具体实现
  Object.assign(_hub, {
  async 'on-confirm'() {
 if (onConfirm) {
 const fn = onConfirm();
 // 当方法返回为 Promise
 if (fn && fn.then) {
 try {

  props.loading = true;
   await fn;
   props.loading = false;
  _closeModal();
 } catch (err) {
  // 发生错误时,不关闭弹框
  console.error(err);
  props.loading = false;
 }
 } else {
 _closeModal();
 }
} else {
 _closeModal();
  }
 },
'on-cancel'() {
 onCancel && onCancel();
 _closeModal();
  }
   });
  }
};
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容