Appearance
Message全局提示组件制作思路 #
做组件库的时候遇到了组件的调用方式是Message.success()在setup
而不是在模板中定义组件的情况 实现方式如下
实现过程 #
1.实现消息的组件模版 #
在组件目录Message定义 Message.vue
vue
<template>
<Transition name="down">
<div class="message" :style="style[type]" v-if="isShow">
<span class="text">{{ text }}</span>
</div>
</Transition>
</template>
<script lang="ts" setup name="Message">
import { onMounted, ref } from 'vue'
export type Type = 'success' | 'error' | 'warning'
const props = defineProps({
type: {
type: String,
required: true
},
text: {
type: String,
required: true
},
duration: {
type: Number,
default: 1500
}
})
// 控制显示 渐入渐出的效果
const isShow = ref(true)
onMounted(()=> {
isShow.value = true
// 当duration时间过后,就隐藏
window.setTimeout(() => {
isShow.value = false
}, props.duration)
})
// 定义一个对象,三种情况的样式,对象key就是类型字符串
const style = {
warning: {
icon: 'icon-warning',
color: '#E6A23C',
backgroundColor: 'rgb(253, 246, 236)',
borderColor: 'rgb(250, 236, 216)'
},
error: {
icon: 'icon-shanchu',
color: '#F56C6C',
backgroundColor: 'rgb(254, 240, 240)',
borderColor: 'rgb(253, 226, 226)'
},
success: {
icon: 'icon-queren2',
color: '#67C23A',
backgroundColor: 'rgb(240, 249, 235)',
borderColor: 'rgb(225, 243, 216)'
}
}
</script>
<style scoped lang="less">
.message {
width: 300px;
height: 50px;
position: fixed;
z-index: 9999;
left: 50%;
margin-left: -150px;
top: 25px;
line-height: 50px;
padding: 0 25px;
border: 1px solid #e4e4e4;
background: #f5f5f5;
color: #999;
border-radius: 4px;
i {
margin-right: 4px;
vertical-align: middle;
}
.text {
vertical-align: middle;
}
}
.down {
&-enter {
&-from {
transform: translate3d(0, -75px, 0);
opacity: 0;
}
&-active {
transition: all 0.5s;
}
&-to {
transform: none;
opacity: 1;
}
}
&-leave {
&-from {
transform: none;
opacity: 1;
}
&-active {
transition: all 0.5s;
}
&-to {
transform: translate3d(0, 75px, 0);
opacity: 0;
}
}
}
</style>
我们调用消息组件不可能在模版中定义 <Message type="success" text="消息" v-if="isShow"></Message>
而是通过在setup中调用Message.success("成功")
来控制消息组件的显示隐藏
那如何实现Message.success的组件调用方式? #
2. 实现message的api的调用 #
在组件目录Message下定义 index.js
js
import { h, render } from "vue"
import MyMessage from "./index.vue"
// 往body中系只能一个盒子,目的是同时只能出现一次消息弹层
function Message({ type, text, duration = 2000 }) {
// 如果要像elui一样点击就创建 并且多次点击创建多个可以帮创建放在里面
// 往body中新增一个盒子 =>为了代码中
const divContainer = document.createElement('div')
divContainer.classList.add('message-container')
document.body.appendChild(divContainer)
// 1. 创建虚拟dom
// const vNode = h(MyMessage, { type, text })
const vNode = h(MyMessage, { type, text })
// 2. 动态render
render(vNode, divContainer)
// 定义延时器
let timer
if(timer){
clearTimeout(timer)
}
// 3. 开启一个延时器,到时间就隐藏
timer = setTimeout(() => {
render(null, divContainer)
}, duration)
}
Message.success = (text, duration=1500) => {
Message({
text,
type: 'success',
duration
})
}
Message.error = (text, duration=1500) => {
Message({
text,
type: 'error',
duration
})
}
Message.warning = (text, duration=1500) => {
Message({
text,
type: 'warning',
duration
})
}
console.log(Message);
export default Message
这样我们的组件的api调用就完成了
参考 #
仿element-ui之Message提示组件 - 掘金 (juejin.cn)
实战:用 Vue3 实现一个 Message 消息组件 - 掘金 (juejin.cn)
前端通用组件开发笔记 - Toast 轻提示、Message 全局提示 - 掘金 (juejin.cn)