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库组件的全部功能。

相关推荐
TimelessHaze5 分钟前
🔥 一文掌握 JavaScript 数组方法(2025 全面指南):分类解析 × 业务场景 × 易错点
前端·javascript·trae
jvxiao37 分钟前
搭建个人博客系列--(4) 利用Github Actions自动构建博客
前端
袁煦丞1 小时前
SimpleMindMap私有部署团队脑力风暴:cpolar内网穿透实验室第401个成功挑战
前端·程序员·远程工作
li理1 小时前
鸿蒙 Next 布局开发实战:6 大核心布局组件全解析
前端
EndingCoder1 小时前
React 19 与 Next.js:利用最新 React 功能
前端·javascript·后端·react.js·前端框架·全栈·next.js
li理1 小时前
鸿蒙 Next 布局大师课:从像素级控制到多端适配的实战指南
前端
前端赵哈哈1 小时前
Vite 图片压缩的 4 种有效方法
前端·vue.js·vite
Nicholas681 小时前
flutter滚动视图之ScrollView源码解析(五)
前端
电商API大数据接口开发Cris1 小时前
Go 语言并发采集淘宝商品数据:利用 API 实现高性能抓取
前端·数据挖掘·api
风中凌乱的L1 小时前
vue 一键打包上传
前端·javascript·vue.js