自学内容网 自学内容网

在vue3中使用tsx结合render封装一个项目内通用的弹窗组件

场景: 在大屏项目中经常需要用到弹窗的需求,通常一个项目内弹窗的样式是一致的,如果不封装一个弹窗组件hook的话,每一个弹窗都需要单独封装为一个组件然后再放到项目layout的最外层,再通过store全局判断哪个弹窗显示,这样显然很麻烦,所以现在我封装一个通过tsx及render渲染弹窗组件的hooks,需要使用弹窗就调用hooks传入弹窗组件就行

思路: 首先封装一个弹窗基础框架组件(包括弹窗外观样式及关闭和底部操作等事件)提供插槽展示每个弹窗的内容部分,然后在需要用到一个弹窗时封装这个弹窗的内容组件,然后调用hooks传入这个内容组件就可以打开弹窗(在封装的hooks中会通过render将弹窗基础组件和内容组件合并渲染出来生成一个元素节点,再通过document.body.appendChild将生成的元素节点挂载到页面中以实现弹窗的展示效果)

代码步骤:

(1)封装弹窗组件框架

<template>
  <el-dialog class="models-default" v-bind="props" :model-value="visible">
    <slot></slot>
    <!-- 弹窗头部插槽 -->
    <template v-if="$slots.header" #header>
      <slot name="header"></slot>
    </template>
    <!-- 弹窗尾部插槽 -->
    <template v-if="isFooter" #footer>
      <!-- 自定义尾部插槽组件显示 -->
      <template v-if="$slots.footer">
        <slot name="footer"></slot>
      </template>
      <!-- 默认尾部按钮列显示 -->
      <template v-else>
        <el-button @click="visible = false">{{ props.cancelBtnText }}</el-button>
        <el-button v-for="(item, index) in footerButtons" :key="index" :icon="item.icon" :type="item.type"
          @click="() => btnClickHandle(item)">{{ item.name }}</el-button>
        <el-button type="primary" @click="confirmHandle">{{ props.confirmBtnText }}</el-button>
      </template>
    </template>
  </el-dialog>
</template><script lang="ts" setup>
import { ref, getCurrentInstance } from 'vue'
import { ElDialog, ElButton, type DialogProps, type ButtonProps } from 'element-plus'const emits = defineEmits(['update:modelValue', 'confirm', 'beforeClose'])
// 传入的弹窗props
const props = withDefaults(defineProps<BProps>(), {
  footerButtons: () => [], //底部按钮
  confirmBtnText: '确认',  //底部确认按钮文字
  cancelBtnText: '取消',   //底部取消按钮文字
  isFooter: true,   //是否展示底部按钮
  showClose: true,  //是否展示关闭图标
  title: '弹窗名称', //弹窗标题
})const visible = ref(true)const instance = getCurrentInstance()const closeModel = (value = false) => {
  visible.value = value
}const confirmHandle = () => {
  emits('confirm', instance?.proxy?.$refs[props.contentRef], closeModel)
}const btnClickHandle = (item: FooterButtons) => {
  // item.onClick && item?.onClick(instance?.proxy?.$refs[props.contentRef], closeModel)
  item.onClick?.(instance?.proxy?.$refs[props.contentRef], closeModel)
}
</script><script lang="ts">
// 约束底部按钮
export interface FooterButtons {
  icon?: string
  name?: string
  type?: ButtonProps['type']
  onClick?: (contentInstance: any, done: () => void) => void
}
// ts接口约束弹窗接收外部的props
interface BProps extends Partial<DialogProps> {
  footerButtons?: FooterButtons[]
  confirmBtnText?: string
  cancelBtnText?: string
  isFooter?: boolean
  contentRef: string
}
</script>
<style lang="scss">
 // 省略样式
</style>

(2)封装tsx渲染弹窗组件的hook(通过tsx将弹窗框架以及传入的弹窗内容组件结合并渲染到页面上)


import type { ComponentInternalInstance, Ref } from 'vue'
import { h, render, onUnmounted, ref } from 'vue'
import { type DialogProps } from 'element-plus'
import type { JSX } from 'vue/jsx-runtime'
// 弹窗框架组件
import Model, { type FooterButtons } from '@/components/models/index.vue'
import { ElConfigProvider } from 'element-plus'
import zhCn from 'element-plus/es/locale/lang/zh-cn'type Content = Parameters<typeof h>[0] | string | JSX.Element
​
interface UseModelProps extends Partial<DialogProps> {
  isFooter?: boolean // 是否展示底部
  footerButtons?: Array<FooterButtons> // 底部按钮(除取消和保存按钮之外的按钮)
  onConfirm?: (contentInstance: any) => void
  onClosed?: () => void
  onOpened?: () => void
}// 弹窗插槽暴露
interface ElDialogSlots {
  header?: () => JSX.Element
  footer?: () => JSX.Element
}interface ContentPropsType {
  ref: string
  [key: string]: any
}interface Options {
  props: UseModelProps // model props
  slots?: ElDialogSlots // 弹窗插槽对象
  contentProps?: ContentPropsType // 弹窗主体插入逐渐props
}/**
 * 窗体模块hooks
 * @param {Object} content 设置弹窗主体组件对象
 * @param {Object} options 配置信息
 * @param {Object} options.props 窗体组件props,继承element-puls dialog组件的所有props
 * @param {Boolean} options.props.isFooter 是否显示底部内容
 * @param {Array} options.props.footerButtons 底部按钮添加
 * @param {Function} options.props.onConfirm 确认按钮回调
 * @param {Function} options.props.onClosed 弹窗关闭回调
 * @param {Function} options.props.onOpened 弹窗打开回调
 * @param {Function} options.slots 弹窗插槽对象
 * @param {Object} options.contentProps 弹窗主体插入组件props
 * @returns
 */
export function useModel (content: Content, options?: Options) {
  // 弹窗组件实例
  const modelInstance: Ref<ComponentInternalInstance | null> = ref(null)
  // 弹窗的元素节点
  let fragment: Element | null = null// 关闭并卸载组件
  const closeAfter = () => {
    if (fragment) {
      render(null, fragment as unknown as Element) // 卸载组件
      fragment.textContent = '' // 清空文档片段
      fragment = null
    }
    modelInstance.value = null
  }
  // 关闭弹窗
  function close () {
    if (modelInstance.value) modelInstance.value.props.modelValue = false
  }
 // 核心代码
  function open () {
    // 打开弹窗前判断如果存在当前弹窗实例就销毁掉重新创建
    if (modelInstance.value) {
      close()
      closeAfter()
    }
    const { props = {}, slots = {}, contentProps = { ref: 'content' } } = options ?? {}
    // 创建元素节点
    fragment = document.createDocumentFragment() as unknown as Element
    // tsx组件内容(将弹窗框架组件和传入的内容组件整合)
    const vNode = (
      <ElConfigProvider locale={zhCn} size="small" zIndex={3000}>
        {/* 将传入的props弹窗配置信息都传入到弹窗框架组件中 */}
        <Model align-center {...props} modelValue={true} contentRef={contentProps.ref}>
          {{
            {/* 弹窗内容组件 */}
            default: () => <content {...contentProps}></content>,
            ...slots,
          }}
        </Model>
      </ElConfigProvider>
    )
    // render将tsx生成的组件vNode渲染到元素节点上
    render(vNode, fragment)
    // 弹窗实例
    modelInstance.value = vNode.component
    // 将弹窗的元素节点挂载到页面上
    document.body.appendChild(fragment)
  }onUnmounted(() => {
    close()
  })
  
  // 将外部需要用到的方法和变量暴露出去
  return {
    open, // 打开弹窗
    close, // 关闭弹窗
    closeAfter, // 销毁弹窗
    modelInstance, // 弹窗实例
  }
}

(3)在需要用弹窗组件的地方使用封装的hooks


// 弹窗hooks
import { useModel } from '@/hooks/useModel'
// 弹窗内容组件
import ShopDetail from '../detailDialog/ShopDetail.vue';const showDetail = () => {
  // 传入弹窗内容组件及弹窗组件需要用到的props参数
  const { open, closeAfter } = useModel(ShopDetail, {
    // 弹窗框架组件props
    props: {
      title: '店铺详情',
      width: '900px',
      isFooter: false,
      // 关闭弹窗回调
      onClosed: () => {
        closeAfter()
      }
    },
    // 弹窗内容组件props
    contentProps: {
      ref: 'detail',
      dataInfo: props.markerData
    },
  })
  // 打开弹窗
  open()
}

原文地址:https://blog.csdn.net/minusing/article/details/144770580

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!