文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

怎么封装一个更易用的Dialog组件

2023-06-30 14:28

关注

这篇“怎么封装一个更易用的Dialog组件”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“怎么封装一个更易用的Dialog组件”文章吧。

场景

在项目中,我们经常会遇到使用弹窗的场景,但有时组件库自带的弹窗不能满足我们的需求,需要我们自己封装,这时我们如何去自定义一个更加方便调用的弹窗?

搭建环境

首先我们需要搭建一个Vue3+ts的环境。

用vite的官方模板:

yarn create vite demo-app --template vue-ts

进入并安装依赖

cd demo-appyarn

依赖安装完成后启动app

yarn dev

创建组件

先在src/components目录下创建MyDialog.vue,搭建一个组件的基本框架

<script lang="ts" setup>import { ref, reactive } from "vue";defineProps({  message: {    type: String,    default: "",  },  title: {    type: String,    default: "",  },});const emits = defineEmits<{  (e: "confirm"): void;  (e: "close"): void;}>();const visible = ref(true);function clickConfirm() {  console.log("确认");  emits("confirm");}function clickClose() {  console.log("取消");  emits("close");}</script><template>  <div class="wrap" v-if="visible">    <div class="container">      <div class="title">{{ title }}</div>      <div class="content">        <div>{{ message }}</div>      </div>      <div class="controll">        <button @click="clickConfirm">确认</button>        <button @click="clickClose">取消</button>      </div>    </div>  </div></template><style scoped>.wrap {  position: absolute;  top: 0;  left: 0;  background: rgba(15, 15, 15, 0.5);  width: 100%;  height: 100%;}.container {  position: absolute;  top: 50%;  left: 50%;  transform: translate(-50%, -50%);  min-width: 300px;  min-height: 200px;  padding: 10px;  background: white;  display: flex;  flex-direction: column;}.content {  flex: 1;  padding: 10px;  text-align: left;}.title {  min-height: 30px;}.controll {  display: flex;  width: 100%;  justify-content: space-around;}</style>

创建调用组件的hook函数

在src目录下创建hooks目录,然后再hooks目录下创建useMyDialog.ts.

函数调用组件我们需要:

import { createVNode, render, ComponentPublicInstance } from "vue";export default function useMyDialog(option?: any) {  const props = {    ...option,  };  const vm = createVNode(MyDialog, props);  const container = document.createElement("div");  render(vm, container);  document.querySelector("#app")?.appendChild(container.firstElementChild!);}

ps:

container.firstElementChild!中的!表示container.firstElementChild不为null或者undefined

接下来我们在App.vue中测试一下

<script setup lang="ts">import useMyDialog from "./hooks/useMyDialog";function showDialog() {  useMyDialog({    message: "test1",    onClose: () => {      console.log("self");    },  });}</script><template>  <button @click="showDialog">显示Dialog</button></template>

怎么封装一个更易用的Dialog组件

Dialog的缓存、隐藏

隐藏

我们需要将close返回出去,这样我们就可以手动调用close函数关闭Dialog.

在useMyDialog.ts中添加

import { ComponentPublicInstance,VNode } from "vue";export default function useMyDialog(option?: any) {  const userCloseFn = option?.onClose;  props.onClose = () =&gt; {    close();    userCloseFn ?? userCloseFn();  };  function close(vm: VNode) {    (      vm.component!.proxy as ComponentPublicInstance&lt;{ visible: boolean }&gt;    ).visible = false;  }  return {    close: close.bind(null, vm),  }}

缓存

怎么封装一个更易用的Dialog组件

现在每次点击显示Dialog按钮时都会创建一个新的组件实例,这不是我们的预期,所以我们需要将组件进行缓存.

在useMyDialog.ts中添加

import { ComponentPublicInstance } from 'vue'const instances: any[] = [];export default function useMyDialog(option?: any) {  const tempVm: any = instances.find(    (item) =>      `${item.vm.props?.message ?? ""}` === `${(option as any).message ?? ""}`  );  if (tempVm) {    (      tempVm.vm.component!.proxy as ComponentPublicInstance<{        visible: boolean;      }>    ).visible = true;    return {      close: close.bind(null, tempVm.vm),    };  }}

完整代码

src/hooks/useMyDialog.ts

import { createVNode, render, ComponentPublicInstance, VNode } from "vue";import MyDialog from "../components/MyDialog.vue";const instances: any[] = [];export default function useMyDialog(option?: any) {  const props = {    ...option,  };  const userCloseFn = option?.onClose;  props.onClose = () => {    close(vm);    userCloseFn ?? userCloseFn();  };  function close(vm: VNode) {    (      vm.component!.proxy as ComponentPublicInstance<{ visible: boolean }>    ).visible = false;  }  const tempVm: any = instances.find(    (item) =>      `${item.vm.props?.message ?? ""}` === `${(option as any).message ?? ""}`  );  if (tempVm) {    (      tempVm.vm.component!.proxy as ComponentPublicInstance<{        visible: boolean;      }>    ).visible = true;    return {      close: close.bind(null, tempVm.vm),    };  }  const vm = createVNode(MyDialog, props);  const container = document.createElement("div");  render(vm, container);  document.querySelector("#app")?.appendChild(container.firstElementChild!);  instances.push({ vm });  return {    close: close.bind(null, vm),  };}

以上就是关于“怎么封装一个更易用的Dialog组件”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯