如何写一个全局的 Notice 组件?
下面将会实现这样的效果:
组件动态创建脚本:
import Vue from "vue";
import Notice from "@/components/Noticer/Notice.vue"; function create(Component, props) {
// 先建立实例
const vm = new Vue({
render(h) {
//h就是createElement,它返回VNode
return h(Component, { props });
},
}).$mount();
// 手动挂载 // 判断是否存在container,如果不存在则先创建
let container;
container = document.querySelector(".noticer-container");
if (container == null) {
container = document.createElement("div");
container.classList.add("noticer-container");
container.style.position = "fixed";
container.style.top = "50px";
container.style.right = "0px";
container.style.overflow = "hidden";
container.style.zIndex = 9999; document.body.appendChild(container);
} container.appendChild(vm.$el); //销毁方法
const comp = vm.$children[0];
comp.remove = function () {
container.removeChild(comp["$el"]);
vm.$destroy();
};
comp.show();
return comp;
} Vue.prototype.$notice = {
error: function (props) {
create(Notice, Object.assign(props, { type: "error" }));
},
info: function (props) {
create(Notice, Object.assign(props, { type: "info" }));
},
success: function (props) {
create(Notice, Object.assign(props, { type: "success" }));
},
warn: function (props) {
create(Notice, Object.assign(props, { type: "warn" }));
},
};这里有一些值得注意的地方:
- container: 的作用是notice的容器,它可以用于定位notice在页面的哪里展示,注意notice不应该随页面卷动,因此其定位是
fixed, 而之所以设定为overflow:hidden的原因则是,notice 在出现和移除的时候,发生的动画偏移,会让页面出现横向滚动条。为了避免重复创建container, 这里做了一个判断逻辑。然后所有动态生成的notice实例dom都会通过appendChild添加到这个容器。 - 在移除的时候, 移除的是
vm.$children[0]["$el"], 原因是,Notice 模板的实现中,外层套了一个 transition , 而这个transition是并不会渲染dom的。
- container: 的作用是notice的容器,它可以用于定位notice在页面的哪里展示,注意notice不应该随页面卷动,因此其定位是
创建Notice组件模板:
<template>
<transition
enter-active-class="animate__animated animate__slideInRight"
leave-active-class="animate__animated animate__slideOutRight"
@after-leave="afterLeave"
>
<div v-if="isShow" class="notice__root">
<div :class="`notice-type-${type}`" class="noticer">
{{
type === "error"
? "🍓"
: type === "success"
? "🍀"
: type === "warn"
? "🍋"
: "🐳"
}}
: {{ message }}
</div>
</div>
</transition>
</template>
<script>
export default {
props: {
title: {
type: String,
default: "",
},
message: {
type: String,
default: "",
},
time: {
type: Number,
default: 1000,
},
type: {
type: String,
},
},
data() {
return {
isShow: false,
};
},
methods: {
show() {
this.isShow = true;
setTimeout(this.hide, this.time);
},
hide() {
this.isShow = false;
},
afterLeave() {
this.remove();
},
},
};
</script>
<style lang="less" scoped>
@error: rgb(255, 30, 30);
@warn: rgb(240, 192, 0);
@success: rgb(0, 144, 74);
@info: rgb(0, 80, 218); @errorBg: rgb(255, 208, 208);
@warnBg: rgb(255, 245, 207);
@successBg: rgb(210, 255, 233);
@infoBg: rgb(203, 222, 255); .notice__root {
user-select: none;
padding: 5px 50px 5px 5px;
} .noticer {
padding: 5px 20px;
margin: 10px 0px;
// margin-right: 50px;
border-radius: 8px;
font-size: 16px;
width: auto;
min-width: 280px;
max-width: 300px;
word-break: break-all;
text-align: center;
box-sizing: border-box;
}
.notice-type-error {
color: @error !important;
border: 2px solid @error;
box-shadow: 1px 1px 5px 2px @errorBg;
background-color: @errorBg; // border: 1px solid red;
}
.notice-type-warn {
color: @warn !important;
border: 2px solid @warn;
background-color: @warnBg;
box-shadow: 1px 1px 5px 2px @warnBg;
}
.notice-type-success {
color: @success !important;
border: 2px solid @success;
background-color: @successBg;
box-shadow: 1px 1px 5px 2px @successBg;
}
.notice-type-info {
color: @info !important;
border: 2px solid @info;
background-color: @infoBg;
box-shadow: 1px 1px 5px 2px @infoBg;
}
</style>
在 main.js 中引入执行该脚本即可
import Vue from "vue";
import App from "./App.vue";
import "animate.css";
import "@/components/Noticer/NotificationBanner.js"; new Vue({
render: (h) => h(App),
}).$mount("#app");
代码中使用实例:
if (!this.nickname) {
this.$notice.error({
message: "好汉!姓甚名谁?",
time: 3000,
});
} else {
this.showModal = false;
this.$notice.info({
message: this.nickname + "来了!!!",
time: 3000,
});
}
动态创建组件的执行逻辑:
当在使用的时候:
this.$notice.error({
message: "好汉!姓甚名谁?",
time: 3000,
});
上方代码触发,实际上会触发 NotificationBanner.js 中的 create函数,该函数创建了一个notice 的组件实例,并在实力上添加了一个remove 方法,然后直接触发组件中的 show 方法。
notice 模板实例中:
methods: {
show() {
this.isShow = true;
setTimeout(this.hide, this.time);
},
hide() {
this.isShow = false;
},
afterLeave() {
this.remove();
},
},
show 方法执行,除了展示 notice,立即设定一个延时函数执行 hide 方法。
hide 方法执行, vue 提供的 transition 钩子 afterleave() 会在移除动画执行完毕后触发。 这时候,去触发 remove 方法,将该notice 组件实例移除。
如何写一个全局的 Notice 组件?的更多相关文章
- 手写一个简单的starter组件
spring-boot中有很多第三方包,都封装成starter组件,在maven中引用后,启动springBoot项目时会自动装配到spring ioc容器中. 思考: 为什么我们springBoot ...
- Tsx写一个通用的button组件
一年又要到年底了,vue3.0都已经出来了,我们也不能一直还停留在过去的js中,是时候学习并且在项目中使用一下Ts了. 如果说jsx是基于js的话,那么tsx就是基于typescript的 废话也不多 ...
- python安装及写一个简单的验证码组件(配合node)
1.安装Python 到官网下载响应系统的版本(这里以windows为例):https://www.python.org/downloads/windows/ 然后就是不断地"下一步&quo ...
- Vue学习—Vue写一个图片轮播组件
1.先看效果: 熟悉的图片轮播,只要是个网站,百分之90以上会有个图片轮播.我认为使用图片轮播. 第一可以给人以一种美观的感受,而不会显得网站那么呆板, 第二可以增加显示内容,同样的区域可以显示更多内 ...
- 用vue2.x注册一个全局的弹窗alert组件
一.在实际的开发当中,弹窗是少不了的,默认系统的弹窗样式太丑,难以满足项目的实际需求,所以需要自己定义弹窗组件,把弹窗组价定义为全局的,这样减少每次使用的时候引入麻烦,节省开发时间.本文将分享如何定义 ...
- 用vue2.x注册一个全局的弹窗alert组件、confirm组件
一.在实际的开发当中,弹窗是少不了的,默认系统的弹窗样式太丑,难以满足项目的实际需求,所以需要自己定义弹窗组件,把弹窗组价定义为全局的,这样减少每次使用的时候引入麻烦,节省开发时间.本文将分享如何定义 ...
- react全局的公共组件-------弹框 (Alert)
最近研究react,发现写一个组件很容易,但是要写一个全局的好像有点麻烦.不能像VUE一样,直接在原型上面扩展,注册全局组件 下面实现一个弹框,只需要引入之后,直接调用方法即可,不需要写入组件 给出我 ...
- 如何优雅的写一个Vue 的弹框
写Vue或者是react 都会遇见弹框的问题.也尝试了多种办法来写弹框,一直都不太满意,今天特地看了一下 Element UI 的源码,模仿着写了一个简易版. 大概有一下几个问题: 1.弹框的层级问题 ...
- 从零开始写一个npm包,一键生成react组件(偷懒==提高效率)
前言 最近写项目开发新模块的时候,每次写新模块的时候需要创建一个组件的时候(包含组件css,index.js,组件js),就只能会拷贝其他组件修改名称 ,但是写了1-2个后发现效率太低了,而且极容易出 ...
随机推荐
- 什么是3D建模?
一.3D建模是什么 将所见所想用立体三维的方式通过计算机技术表现出来,这就是3D建模,比如你看到了一个可爱呆萌的卡通人物又或是华丽酷炫的变形金刚,总之就是你认为美好的立体实物 你想将它用计算机技术完美 ...
- FreeRTOS学习记录--任务创建函数详解
开局一张图.一步一步分析就好. (一)什么是任务? 在多任务系统中,我们按照功能不同,把整个系统分割成一个个独立的,且无法返回的函数,这个函数我们称为任务:任务包含几个属性:任务堆栈,任务函数.任务控 ...
- 学习打卡day12&构建之法阅读笔记第一篇
今天浅读了<构建之法>的前四章,稍微有一些个人的见解与感受 第一点即是开篇提及到的算法与数据结构这门学科开设的必要,大二上学期学习了这门课程,就我个人目前接触到的层面来看,几乎可以说用不太 ...
- 2021.08.01 P4359 伪光滑数(二叉堆)
2021.08.01 P4359 伪光滑数(二叉堆) [P4359 CQOI2016]伪光滑数 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意: 若一个大于 11 的整数 MM ...
- 震惊!<string.h>、<cstring>和<string>竟然可以这么用!
为什么有这么多string相关的头文件呢,小编秦始皇今天带大家看一下: 1.[string.h] 定义如下:"C语言标准库中一个常用的头文件,在使用到字符数组时需要使用.[strin ...
- XCTF练习题---MISC---Erik-Baleog-and-Olaf
XCTF练习题---MISC---Erik-Baleog-and-Olaf flag:flag{#justdiffit} 解题步骤: 1.观察题目,下载附件 2.拿到手以后发现是一个没有后缀名的文件, ...
- 关于JS精度缺失问题
问题描述 在Java后端传一个比较大的Long值的时候 前端接收值的时候会出现精度的缺失: 解决办法 添加一个转换类 点击查看代码 public class JacksonObjectMapper e ...
- 一文了解RPC框架原理
点击上方"开源Linux",选择"设为星标" 回复"学习"获取独家整理的学习资料! 1.RPC框架的概念 RPC(Remote Proced ...
- 按照 Promise/A+ 规范逐行注释并实现 Promise
0. 前言 面试官:「你写个 Promise 吧.」 我:「对不起,打扰了,再见!」 现在前端越来越卷,不会手写 Promise 都不好意思面试了(手动狗头.jpg).虽然没多少人会在业务中用自己实现 ...
- position与float
position:fixed/absolute和float的关系:元素设置position:absolute / fixed后,float属性是没有效果的.对于position: absolute元素 ...