【vue3 封装一个弹框】封装一个方法通过命令方式的实现弹框,带你了解h,render

【vue3 封装一个弹框】封装一个方法通过命令方式的实现弹框,带你了解h,render

五月 21, 2022 评论 10 阅读 1105 点赞 0 收藏 0

请输入...

需求描述

  1. 点击按钮 让弹框显示
  2. 封装一个方法通过命令方式的调用让弹框展示
    this.$message()
    alert() 在任何你想要显示弹框的条件下只要调用方法就可以展示弹框

思路分析

  1. 核心思路 把本来自动渲染组件方式变成一个手动渲染
  2. 学习一下vue 暴露出来和渲染相关的api
    h render
    h: 把一个组件变成一个VNode
    render: 把VNode 渲染成一个真实dom 节点
  3. 把已经写好的 Alert 组件调用 h 函数 变成VNode
  4. 准备一个渲染挂载的节点
  5. 调用 render api 把 VNode 挂载到准备好的节点上 完成渲染

代码实现

组件代码

tips.vue

<script setup>
defineProps({
  msg: String
})

</script>

<template>
  <div class="tips">{{msg}}</div>
</template>


<style scoped>
.tips {
  position: fixed;
  top: 50%;
  left: 50%;
  transform: translate(-50%,-50%);
  padding: 8px 16px;
  border-radius: 4px;
  font-size: 12px;
  background-color: #f0f9eb;
  color: #67c23a;
  overflow: hidden;
  opacity: 1;
  display: flex;
  align-items: center;
  transition: opacity .2s;
  user-select: none;
}
</style>

tips.js

// 封装函数调用alert
import TipsHtml from './src/tips.vue'
import { h,render } from 'vue'

function clickClose (e) {
    console.log(e,'clickClose');
    e.target.parentNode.remove()
}

function tips (msg="tips") {
    // 1. 调用h把组件变成组件虚拟dom VNode
    let id = 'tips_'+new Date().getTime()
    const VNode = h(TipsHtml,{ 
        msg,
        onClick: clickClose,
     })// js对象  
    // 2. 准备挂载节点
    const container = document.createElement('div')
    container.id = id
    document.body.appendChild(container)
    // 3. 渲染虚拟dom到真实dom
    render(VNode,container)
}

export { tips }

调用组件

index.vue



<template>
<button @click="handleOpen" > open Tips </button>
</template>

<script setup>
// This starter template is using Vue 3 <script setup> SFCs
// Check out https://vuejs.org/api/sfc-script-setup.html#script-setup
import { tips } from '@/components/Tips/index'
const handleOpen = () => tips("内容内容内容内容内容内容内容内容")
</script>

<style>
#app {
  font-family: Avenir, Helvetica, Arial, sans-serif;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  text-align: center;
  color: #2c3e50;
  margin-top: 60px;
}
</style>

更多内容

*
*
*