引言

页面引用弹出框组件是经常碰见的需求,如果强行将弹出框组件放入到页面中,虽然功能上奏效但没有实现组件与页面间的解耦,非常不利于后期的维护和功能的扩展.下面举个例子来说明一下这种做法的弊端.

  
@click="openModal()">点击 :is_open="is_open" @close="close()"/>

import Modal from "../components/Modal/Modal";//外部引入的弹出框组件export default { components: { Modal, }, data(){ return { is_open:false //控制弹出框关闭或打开 } }, methods: { openModal() { //显示弹出框 this.is_open = true; }, close(){ //子组件触发的事件,关闭弹出框 this.is_open = false; } },};

Modal是外部引入的弹出框组件,父组件通过is_open来控制弹出框的隐藏和显示.仔细分析上述结构存在的问题如下.

•Modal组件被硬编码,强行在父组件的components里面注册并在父组件的模板中渲染.设想一下,一个弹出框组件就需要在父组件中写一次,5个弹出框也都要在父组件的模板里写五个.这样会让父组件的页面结构变的复杂不利于阅读,其次弹出框组件应该与父组件解耦,它不应该写死在父组件的模板中.•父组件需要单独设置一个状态is_open来控制弹出框的显示和隐藏,假如父组件需要引入多个弹出框,那势必也要定义多个状态来对弹出框进行控制.

为了实现弹出框和父组件的解耦,最理想的方式是运用函数式编程的思想,在父组件内只需要调用一个函数就可以让弹出框显示出来,接下来看一下如何实现.

弹出框组件的处理

我们接下来实现一个代码十分简单但功能强大的工具函数,借助它就可以将弹出框组件封装起来.如果父组件需要使用哪个弹出框组件直接调用函数就能轻松显示或者隐藏.

实现

import Vue from 'vue';export const createModal = (Component, props) => {  const vm = new Vue({    render: (h) =>      h(Component, {        props,      }),  }).$mount();  document.body.appendChild(vm.$el);  const ele = vm.$children[0];  ele.destroy = function() {    vm.$el.remove();    ele.$destroy();    vm.$destroy();  };  return ele;};

Component就是父组件调用的弹出框组件,在这里作为参数传入.props是最终传递给弹出框组件内部的propsnew 一个 Vue实例,render属性对应的函数里,h的作用是将弹出框组件变成虚拟dom•$mount一定要调用,它会将虚拟dom转换成真实的dom元素•vm.$el就是对应到传入的弹出框组件Component所渲染的真实dom,将它挂载到body下面,此时页面就会显示出弹出框•光显示出弹出框还不够,我们还需要给弹出框组件创建一个销毁方法destroy,其中vm.$children[0]对应的就是弹出框组件的vue实例,可以调用destroy方法销毁.最后将该实例返回供外部调用,外部通过该实例就可以调用弹出框组件内部的属性和方法.

应用

作为测试Demo,弹出框组件结构如下,模板内容十分简单.渲染一个头部标题title和内容content.定义两个方法show()hide()来操作is_open状态来控制弹出框的显示和隐藏.

  
class="modal" v-if="is_open">

class="content">

class="close" @click="hide()">close

class="title">{{ title }}

{{ content }}

export default { props: ["title", "content"], data() { return { is_open: false, }; }, methods: { show() { this.is_open = true; }, hide() { this.is_open = false; }, },};lang="scss" scoped>.modal { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background-color: rgba(0, 0, 0, 0.6); .content { width: 200px; height: 200px; background-color: #fff; margin: 0px auto; margin-top: 200px; text-align: center; font-size: 14px; color: #333; padding: 5px; .title { margin-bottom: 20px; font-size: 16px; } .close { text-align: right; } }}

页面组件

  
class="test-v2"> @click="openModal()">点击

import Modal from "../../components/Modal/Modal";import { createModal } from "../../util/Modal";export default { methods: { openModal() { this.ele = createModal(Modal, { title: "弹出框", content: "内容", }); this.ele.show(); } },};

页面父组件通过调用createModal方法能获取到弹出框组件Modal的实例this.ele.通过this.ele就可以拿到弹出框组件内部的所有属性和方法,包括显示show()和隐藏hide().

•经过上方一改造,实现了弹出框组件和父组件之间的解耦.弹出框组件不需要在父组件中注册和模板内渲染.•如果父组件需要传递数据给弹出框组件,可以借助createModal第二个参数对象,它最终会以props的形式注入到弹出框组件的内部.•show()hide()方法都是弹出框内部定义的,父组件可以直接调用控制其显示隐藏.另外页面销毁时要调用一次this.ele.destroy(),防止内存泄漏.

页面效果

在这里插入图片描述

从最终的dom结构图可以清晰的看到弹出框挂载在body的下面,而非页面组件内部.这样在对弹出框定义一些与css定位相关的样式时就轻松方便的多,不会受到页面组件的影响和干扰.

延伸

通过上面对弹出框的讲解我们还可以在此基础做很多其他的事情,比如对消息提示框的处理.

消息提示框也属于弹出框.最好的实践方式是,只需要写一行代码 Alert("Hello world"),页面上就会立马弹出消息提示 Hello world.效果如下.

在这里插入图片描述

实现

父页面结构如下,调用Alert()函数,页面就会显示提示框.

  
class="test-v2"> @click="alert()">Alert

import { Alert } from "../../util/Modal";export default { methods: { alert() { Alert("Hello world"); }, },};

Alert函数实现如下.

const alert_array = []; //用来存储弹出框的实例export const Alert = (msg, duration = 3000) => {  let top = 100; //默认距离顶部100px  if (alert_array.length > 0) {    const index = alert_array.length;    top = top + index * 50;  }  const ele = createModal(AlertComponent, {    title: msg,    top,  });  alert_array.push(ele);  const timer = setTimeout(() => {    clearTimeout(timer);    const index = alert_array.indexOf(ele);    index !== -1 && alert_array.splice(index, 1);    ele.destroy();  }, duration);};

AlertComponent是自定义的消失提示框组件(需要引入),调用createModal()获取每个提示框的实例存储在数组alert_array中.•点击一次按钮出现一个消息提示框,点击第二次按钮时,第二个提示框应该出现在第一个框的下面,因此需要根据数组alert_array动态计算绝对定位的top值,在创建弹出框实例时作为参数传进去.•定时器控制默认3秒后移除弹出框.

AlertComponent消息提示框组件内容如下.初始给top_value赋值this.top - 30,后来在mounted中再将this.top赋值一次,就是为了实现提示框出现时从上往下滑动的动画效果.

  
class="alert-component" :style="{ top: `${top_value}px`, opacity: opacity }" > {{ title }}

export default { props: ["title", "top"], data() { return { top_value: this.top - 30, opacity: 0, }; }, mounted() { const timer = setTimeout(() => { clearTimeout(timer); this.top_value = this.top; this.opacity = 1; }); },};.alert-component { height: 20px; border-radius: 4px; position: absolute; min-width: 300px; left: 50%; transform: translateX(-50%); background-color: #f0f9eb; color: #67c23a; align-items: center; padding: 10px 16px; transition: all 0.25s linear; opacity: 0;}

结尾

借助createModal工具函数,不仅可以做消息提示框,另外包括消息确认框,动态的表单模态框都可以实现进一步的封装简化处理.当弹出框与页面实现解耦后,整体的代码逻辑会变得更加清晰,对后期维护和扩展都有巨大的好处.

modal vue 关闭_Vue弹出框的优雅实践相关推荐

  1. vue 一个组件内多个弹窗_使用vue实现各类弹出框组件

    简单介绍一下vue中常用dialog组件的封装: 实现动态传入内容,实现取消,确认等回调函数. 首先写一个基本的弹窗样式,如上图所示. 在需要用到弹窗的地方中引入组件: import dialogBa ...

  2. layui table 弹出层刷新_layui 关闭open弹出框 刷新table表格页面的方法

    layui 关闭open弹出框 刷新table表格页面的方法 如下所示: 保存后刷新table表格 源码 //弹出框 layer.open({ type: 2, shadeClose: true, s ...

  3. vue element-ui 日期弹出框右侧被遮挡

    问题: 由于项目需要,引入了一个视频控件,日期弹出框被视频控件所遮挡 解决: 日期弹出框默认弹出方式是左对齐,调整为右对齐就不会被遮挡了 <el-date-pickerv-model=" ...

  4. vue点击input框出现弹窗_使用vue实现各类弹出框组件

    简单介绍一下vue中常用dialog组件的封装: 实现动态传入内容,实现取消,确认等回调函数. 首先写一个基本的弹窗样式,如上图所示. 在需要用到弹窗的地方中引入组件: import dialogBa ...

  5. js关闭layui弹出框

    如有index那时非常简单的 layer.close(index) 但如果我们在外部取不到index呢 其实也不难 其中包含的几个函数 layer.closeAll(); //疯狂模式,关闭所有层 l ...

  6. vue+element el-dialog弹出框会变暗解决办法

    问题如图: 解决方法:加入append-to-body

  7. vue点击input框出现弹窗_vue组件实现弹出框点击显示隐藏效果

    本文实例为大家分享了vue实现弹出框点击显示隐藏的具体代码,供大家参考,具体内容如下 效果如下图 由于我的更改密码弹出框是一个组件引用的,所以在一开始是隐藏的,这就需要在当前的页面上对弹出框组件设置v ...

  8. vue 组件弹出框点击显示隐藏

    本案实现的效果如下图 由于我的更改密码弹出框是一个组件引用的,所以在一开始是隐藏的,这就需要在当前的页面上对弹出框组件设置v-show,但是在弹出框显示出来的时候,操作执行完后当前页面的更改按钮已经被 ...

  9. artDialog弹出新页面,保存后关闭弹出框并刷新父页面

    artDialog 弹出框插件,或者说是弹出对话框插件,需要对弹出后的对话框操作?artDialog的控制接口就是用来干这些事情的,这在异步消息操作中非常有用. artDialog功能特性: 自适应内 ...

最新文章

  1. HDU3923-Invoker-polya n次二面体
  2. 学习Python语言 基础语法:变量的基本使用
  3. Anaconda 镜像使用帮助
  4. 查一个字段中字符集超过30的列_详细解读MySQL的30条军规
  5. JavaScript中关于setTimeout和setInterval的使用
  6. 人工智能大脑如何调控智能交通“疏堵”?
  7. [渝粤教育] 中国地质大学 现代控制理论 复习题 (2)
  8. 在 Mac 上的“照片”中如何把文件夹中的相簿分组?
  9. 模块化分析设计(简单的注册登录模块)
  10. Java后端学习路线
  11. 厦门理工学院计算机毕业要求,计算机教学中心-厦门理工学院教务处.PDF
  12. linux下载tftpd服务,Linux tftpd服务安装与配置
  13. BZOJ1299 巧克力棒
  14. 上下相机贴合对位计算公式_相机界的小公主康泰时Contax g1
  15. linux下firefox浏览器的flash版本过低解决方案
  16. 35岁鹅厂员工失业后嚎啕大哭...年轻新鹅分享省钱秘诀,每月除了房租水电,其他吃喝玩乐基本白嫖公司......
  17. 1103 缘分数 – PAT乙级真题
  18. 计算机软件与应用是什么学士,软件工程什么学士
  19. Tomcat热部署方法(3种)

热门文章

  1. AVS解码器在DSP平台上的优化
  2. 面试风云录(02) - 与顶级CTO交手的难忘经验...
  3. Skip宏块与Direct预测模式浅析
  4. H.264的技术优势及其在H.323系统中的应用
  5. 查看 linux 硬件信息:内存、分区、系统、环境变量、防火墙、路由、端口监听、进程、CPU...
  6. 版本控制:集中式(SVN) vs 分布式(GIT)
  7. 解决:If the number of processors is expected to increase from one, then you should configure the numbe
  8. Qt 程序获取程序所在路径、用户目录路径、临时文件夹等特殊路径的方法
  9. sqlserver字符串多行合并为一行
  10. ie7浏览器传输中文的问题