MerchantIndex.vue 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. <template>
  2. <el-dialog :model-value="dialogVisible" @update:model-value="updateDialogVisible" class="custom-dialog">
  3. <div class="dialog-content">
  4. <div class="tabs-section">
  5. <el-tabs v-model="activeTab" tab-position="left">
  6. <el-tab-pane v-for="tab in tabs" :key="tab.name" :label="tab.title" :name="tab.name" />
  7. </el-tabs>
  8. </div>
  9. <div class="component-section">
  10. <component :is="currentComponent" />
  11. </div>
  12. </div>
  13. </el-dialog>
  14. </template>
  15. <script setup>
  16. import { ref, watch, defineAsyncComponent } from 'vue';
  17. const props = defineProps({
  18. dialogVisible: Boolean,
  19. tabs: Array, // 父组件传来的tabs数组,每个tab对象包含title和name属性
  20. components: Object // 父组件传来的组件地址映射,键为tab.name,值为组件路径字符串
  21. });
  22. const emit = defineEmits(['update:dialogVisible']);
  23. const componentMap = props.components;
  24. const activeTab = ref(props.tabs[0]?.name || '');
  25. const currentComponent = ref(null);
  26. watch(activeTab, (newTab) => {
  27. if (componentMap[newTab]) {
  28. const componentPath = componentMap[newTab];
  29. currentComponent.value = defineAsyncComponent(() =>
  30. import(/* webpackChunkName: "[request]" */ componentPath)
  31. );
  32. }
  33. }, { immediate: true });
  34. function updateDialogVisible(value) {
  35. emit('update:dialogVisible', value);
  36. }
  37. </script>
  38. <style scoped>
  39. .custom-dialog .dialog-content {
  40. display: flex;
  41. }
  42. .tabs-section {
  43. flex: 1;
  44. }
  45. .component-section {
  46. flex: 3;
  47. }
  48. </style>