相信普通的vue组件大家都会写,定义 -> 引入 -> 注册 -> 使用,行云流水,一气呵成,但是如果我们今天是要自定义一个弹窗组件呢?

首先,我们来分析一下弹窗组件的特性(需求): 
0. 轻量 –一个组件小于 1Kib (实际打包完不到0.8k) 
1. 一般都是多处使用 –需要解决每个页面重复引用+注册 
2. 一般都是跟js交互的 –无需 在<template>里面写 <toast :show="true" text="弹窗消息"></toast>

今天,我们就抱着上面2个需求点,来实现一个基于vue的toast弹窗组件,下图是最终完成的效果图.

一. 先写一个普通的vue组件

文件位置 /src/toast/toast.vue

<template><div class="wrap">我是弹窗</div> </template> <style scoped> .wrap{ position: fixed; left: 50%; top:50%; background: rgba(0,0,0,.35); padding: 10px; border-radius: 5px; transform: translate(-50%,-50%); color:#fff; } </style>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

二. 在我们需要使用的页面引入组件,方便看效果和错误

<template><div id="app"> <toast></toast> </div> </template> <script> import toast from './toast/toast' export default { components: {toast}, } </script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

三. 实现动态加载组件

可以看到,已经显示出一个静态的弹出层了,接下来我们就来看看如何实现动态弹出.

我们先在 /src/toast/ 目录下面,新建一个index.js, 然后在index.js里面,敲入以下代码(由于该代码耦合比较严重,所以就不拆开一行一行讲解了,改成行内注释)

文件位置 /src/toast/index.js

import vue from 'vue'// 这里就是我们刚刚创建的那个静态组件
import toastComponent from './toast.vue'// 返回一个 扩展实例构造器
const ToastConstructor = vue.extend(toastComponent) // 定义弹出组件的函数 接收2个参数, 要显示的文本 和 显示时间 function showToast(text, duration = 2000) { // 实例化一个 toast.vue const toastDom = new ToastConstructor({ el: document.createElement('div'), data() { return { text:text, show:true } } }) // 把 实例化的 toast.vue 添加到 body 里 document.body.appendChild(toastDom.$el) // 过了 duration 时间后隐藏 setTimeout(() => {toastDom.show = false} ,duration) } // 注册为全局组件的函数 function registryToast() { // 将组件注册到 vue 的 原型链里去, // 这样就可以在所有 vue 的实例里面使用 this.$toast() vue.prototype.$toast = showToast } export default registryToast
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37

附一个传送门 vue.extend 官方文档

四. 试用

到这里,我们已经初步完成了一个可以全局注册和动态加载的toast组件,接下来我们来试用一下看看

  1. 在vue的入口文件(脚手架生成的话是./src/main.js) 注册一下组件

文件位置 /src/main.js

import toastRegistry from './toast/index'// 这里也可以直接执行 toastRegistry()
Vue.use(toastRegistry)
  • 1
  • 2
  • 3
  • 4
  1. 我们稍微修改一下使用方式,把第二步 的引用静态组件的代码,改成如下
<template><div id="app"> <input type="button" value="显示弹窗" @click="showToast"> </div> </template> <script> export default { methods: { showToast () { this.$toast('我是弹出消息') } } } </script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

可以看到,我们已经不需要在页面里面引入注册组件,就可以直接使用this.$toast()了.

五. 优化

现在我们已经初步实现了一个弹窗.不过离成功还差一点点,缺少一个动画,现在的弹出和隐藏都很生硬.

我们再对 toast/index.js 里的showToast函数稍微做一下修改(有注释的地方是有改动的)

文件位置 /src/toast/index.js

function showToast(text, duration = 2000) { const toastDom = new ToastConstructor({ el: document.createElement('div'), data() { return { text:text, showWrap:true, // 是否显示组件 showContent:true // 作用:在隐藏组件之前,显示隐藏动画 } } }) document.body.appendChild(toastDom.$el) // 提前 250ms 执行淡出动画(因为我们再css里面设置的隐藏动画持续是250ms) setTimeout(() => {toastDom.showContent = false} ,duration - 1250) // 过了 duration 时间后隐藏整个组件 setTimeout(() => {toastDom.showWrap = false} ,duration) } 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

然后,再修改一下toast.vue的样式

文件位置 /src/toast/toast.vue

<template><div class="wrap" v-if="showWrap" :class="showContent ?'fadein':'fadeout'">{{text}}</div> </template> <style scoped> .wrap{ position: fixed; left: 50%; top:50%; background: rgba(0,0,0,.35); padding: 10px; border-radius: 5px; transform: translate(-50%,-50%); color:#fff; } .fadein { animation: animate_in 0.25s; } .fadeout { animation: animate_out 0.25s; opacity: 0; } @keyframes animate_in { 0% { opacity: 0; } 100%{ opacity: 1; } } @keyframes animate_out { 0% { opacity: 1; } 100%{ opacity: 0; } } </style>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39

大功告成,一个toast组件初步完成

总结

  1. vue.extend 函数可以生成一个 组件构造器 可以用这个函数构造出一个 vue组件实例
  2. 可以用 document.body.appendChild() 动态的把组件加到 body里面去
  3. vue.prototype.$toast = showToast 可以在全局注册组件
  4. 显示动画比较简单,隐藏动画必须要在隐藏之前预留足够的动画执行时间
  5. 本文源码地址 在这里
  6. 以上都不重要,重要的是 给本文来个star

转载于:https://www.cnblogs.com/xiaocaiyuxiaoniao/p/9449214.html

从零开始徒手撸一个vue的toast弹窗组件相关推荐

  1. 带你从零开始徒手撸光线追踪代码(1)—— Ray Tracing in One Weekend

    前言   开坑一个新系列,由笔者带你不用api,只用最基本的C++语法写光线追踪,本系列可以看做是对<Ray Tracing in One Weekend>的翻译,当然原文虽然是英文,但有 ...

  2. spring boot 自动跳转登录页面_徒手撸一个扫码登录示例工程

    徒手撸一个扫码登录示例工程 不知道是不是微信的原因,现在出现扫码登录的场景越来越多了,作为一个有追求.有理想新四好码农,当然得紧跟时代的潮流,得徒手撸一个以儆效尤 本篇示例工程,主要用到以下技术栈 q ...

  3. vue 定义全局弹框_VUE路由拦截:Vue自定义全局弹窗组件

    前言 在任何一个平台中,如果需要增加用户黏度,除了用户需要的基本内容外,用户登录注册提交信息也是非常重要的一环,可以了解用户基本信息,用户喜欢等. 抛出前后端混合开发外,vue可以轻松的实现路由拦截. ...

  4. vue写一个通用的toast弹窗 toast 弹窗 提示

    效果图 代码 <!DOCTYPE html> <html lang="en"><head><title>弹窗</title&g ...

  5. 撸一个vue的双向绑定

    1.前言 说起双向绑定可能大家都会说:Vue内部通过Object.defineProperty方法属性拦截的方式,把data对象里每个数据的读写转化成getter/setter,当数据变化时通知视图更 ...

  6. 从零开始,创建一个VUE项目,详细图文详解。

    准备工作 1.安装VScode 点击下载 2.安装node 点击下载 3.安装npm 点击下载 一.步骤详解 1.创建项目的文件夹.比如F盘下新建一个名为MyDemo的文件夹,打开VScode进入My ...

  7. vue PC端弹窗组件封装

    在项目中,需要做一个可以任意更改颜色的弹窗,找了一些组件,发现都不是自己想要的,打算自己动手写一下. 先看看效果图 实现的效果主要是可以自由更改按钮文字的大小,背景色等 接下来看看我操作的代码 首先, ...

  8. diff算法_[VUE学习]徒手撸一个虚拟dom+diff算法

    首先先来聊一下,什么是虚拟dom? 为什么要使用虚拟dom? 其实答案很简单,虚拟dom就是位于js层和html层之间的一个层,使用js模拟出一个dom树,然后通过diff算法,来侦测到dom发生更改 ...

  9. 从零开始手撸一个热修复框架

    1.前言 热修复原理,这个一直是这几年来很热门的话题,在项目中使用的话,也基本要么是阿里系或者腾讯系的开源框架.但是作为一个光会使用的程序员是远远不够的.这篇文章会从dex分包的原因,原理,热修复的由 ...

最新文章

  1. 如何在 Fedora 上使用 Podman
  2. 找出如下数组中最大的元素和最小的元素, a[][]={{3,2,6},{6,8,2,10},{5},{12,3,23}}
  3. 成功解决PermissionError: [WinError 5] 拒绝访问 \lib\\site-packages\\h5py\\_errors.cp38-win_amd64.pyd‘
  4. 前端学习(2376):项目初始化
  5. php 返回数组元素函数_php array_values 返回数组的所有值详解及实例
  6. 微软“照片”应用Raw 格式图像编码器漏洞 (CVE-2021-24091)的技术分析
  7. 局域网内通过代理服务器上网的详细设置
  8. 持续集成实践二之Jenkins与Sonar Qube集成
  9. [网安实践II] 实验2. 密码学实验
  10. 【Java】转义字符
  11. 大数据在智慧城市建设中的应用
  12. 计算机电源再利用usb充电器,变废为宝,笔记本旧电源改造的8口USB充电器
  13. 常用数学希腊符号读法
  14. js html显示emoji表情,js emoji表情长度判断
  15. 安徽新科技计算机学校是高中吗,2021最新安徽高中排名一览表 所以高中学校
  16. 『随笔』基本功与招式
  17. appStore 应用审核-加急审核
  18. 阿里云飞天计划体验-阿里云开发者社区
  19. Python升级pip失败解决办法
  20. Android高版本使用DocumentFile读写外置存储的问题

热门文章

  1. java集合迭代器_java集合迭代器
  2. 实现第一个JDBC程序(详细)
  3. 加菲猫软件显示不能链接服务器,sql server 2012 链接服务器不能链接sql server 2000的解决方案...
  4. python 遍历矩阵_Python3算法之十:矩阵旋转
  5. 打印服务器协议,基于TCP/IP协议的嵌入式网络打印服务器设计
  6. oracle客观题题库,oralce题库及答案.doc
  7. PHP系统发布指什么,cis系统是指什么
  8. 华文行楷字帖欣赏_田英章硬笔行书字帖欣赏,600个行书笔法演示,写好字的首选帖...
  9. pandas 如何删掉第一行_Pandas-数据筛选
  10. java bufferedreader类_详解java中的BufferedReader类