vue3如何封装统一的弹窗

在Vue3开发中,我们经常需要对UI组件库进行二次封装以满足项目需求。

下面我将分享一些实用的封装技巧,帮助你创建更灵活、可维护的组件。

初始封装方案的问题

js 复制代码
<template>
  <div ref="myModal" class="custom-modal"></div>
  <a-modal
    v-model:visible="visible"
    centered
    destroyOnClose
    :getContainer="() => $refs.myModal"
    @ok="handleOk"
    @cancel="handleCancel"
    :style="{ width: '560px', ...style }"
    :cancelText="cancelText"
    :okText="okText"
  >
    <slot></slot>
  </a-modal>
</template>

<script setup>
const props = defineProps({
  title: String,
  style: Object,
  cancelText: { type: String, default: "取消" },
  okText: { type: String, default: "确定" }
});

const emits = defineEmits(["handleOk", "handleCancel"]);
const visible = ref(false);

const handleOk = () => emits("handleOk");
const handleCancel = () => emits("handleCancel");

defineExpose({ visible });
</script>

这种封装方式存在明显问题:每次新增需求都需要修改组件代码,导致组件越来越臃肿

优化方案:使用$attrs$slots

1. 利用$attrs传递属性

js 复制代码
<template>
  <div ref="myModal" class="custom-modal"></div>
  <a-modal
    v-model:visible="visible"
    centered
    destroyOnClose
    :getContainer="() => $refs.myModal"
    :style="{ width: '560px' }"
    v-bind="$attrs"
  >
    <template v-for="(_, name) in $slots" #[name]="slotData">
      <slot :name="name" v-bind="slotData || {}"></slot>
    </template>
  </a-modal>
</template>

<script setup>
const visible = ref(false);
defineExpose({ visible });
</script>

2. 支持v-model控制显隐

js 复制代码
<script setup>
const props = defineProps({
  visible: Boolean
});

const emit = defineEmits(['update:visible']);

watch(
  () => props.visible,
  (newVal) => emit('update:visible', newVal)
);
</script>

完整优化后的组件

js 复制代码
<template>
  <div ref="myModal" class="custom-modal"></div>
  <a-modal
    :visible="internalVisible"
    centered
    :getContainer="() => $refs.myModal"
    :style="{ width: '560px' }"
    destroyOnClose
    v-bind="$attrs"
    @update:visible="handleVisibleChange"
  >
    <template v-for="(_, name) in $slots" #[name]="slotData">
      <slot :name="name" v-bind="slotData || {}"></slot>
    </template>
  </a-modal>
</template>

<script setup>
import { ref, watch } from 'vue';

const props = defineProps({
  visible: {
    type: Boolean,
    default: false
  }
});

const emit = defineEmits(['update:visible']);

const internalVisible = ref(props.visible);

watch(
  () => props.visible,
  (newVal) => {
    internalVisible.value = newVal;
  }
);

const handleVisibleChange = (value) => {
  internalVisible.value = value;
  emit('update:visible', value);
};
</script>

<style lang="less" scoped>
.custom-modal {
  :deep(.ant-modal) {
    // 自定义样式
  }
}
</style>

使用示例

js 复制代码
<CustomModal 
  v-model:visible="visible" 
  title="自定义标题" 
  :closable="false"
  :footer="null"
>
  <template #title>
    <h2>自定义标题内容</h2>
  </template>
  
  <p>这是模态框的内容</p>
</CustomModal>

封装技巧总结

  1. 使用$attrs传递属性 - 自动绑定父组件传递所有属性内部组件
  2. 使用$slots动态处理插槽 - 支持所有原生插槽不需要显式声明
  3. 支持v-model - 提供更直观的双向绑定方式
  4. 保持组件简洁 - 避免过度封装,只添加必要的默认值

这种方法使你的封装组件能够:

  • 支持UI库组件所有原生属性事件
  • 支持所有插槽用法
  • 易于维护和扩展
  • 保持与原生组件几乎相同的API

通过这种方式封装组件,你可以在保持项目统一风格的同时,享受到UI库组件的全部功能。

相关推荐
T___T8 小时前
彻底搞懂 CSS 盒子模型 box-sizing:小白也能看懂的布局核心
前端·面试
彭于晏爱编程8 小时前
关于表单,别做工具库舔狗
前端·javascript·面试
空白格978 小时前
Android插件化开发
前端
风中凌乱的L8 小时前
vue canvas标注
前端·vue.js·canvas
拉不动的猪8 小时前
什么是二义性,实际项目中又有哪些应用
前端·javascript·面试
海云前端18 小时前
Webpack打包提速95%实战:从20秒到1.5秒的优化技巧
前端
烟袅8 小时前
LeetCode 142:环形链表 II —— 快慢指针定位环的起点(JavaScript)
前端·javascript·算法
梦6508 小时前
什么是react?
前端·react.js·前端框架
zhougl9968 小时前
cookie、session、token、JWT(JSON Web Token)
前端·json
Ryan今天学习了吗8 小时前
💥不说废话,带你上手使用 qiankun 微前端并深入理解原理!
前端·javascript·架构