Skip to content
本页目录

Message全局提示组件制作思路

image-20230620114406825

demo-doc: 示例仓库 - Gitee.com

做组件库的时候遇到了组件的调用方式是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)

vue2:vue实现全局自定义message提示消息弹窗 - 掘金 (juejin.cn)

JS给函数添加属性_Sfh_11的博客-CSDN博客

js中利用prototype给类添加方法_学习笔记666的博客-CSDN博客