Vue3 + TypeScript,使用祖先传后代模式重构父传子模式

父传子模式

父组件 SampleInput.vue

javascript 复制代码
<script setup lang="ts" name="SampleInput">
import { ref } from "vue";
import type { ApplyBasicInfo, Apply, ApplySample } from "@/interface";
import CommonApplySampleTable from "@/components/common/CommonApplySampleTable.vue";

// 定义数据
// 受理基础信息
const applyBasicInfo = ref<ApplyBasicInfo | null>(null)
// 受理样品表格数据
const applySampleTableData = ref<ApplySample[]>([]);
// 滚动到离顶部表头的距离
const applySampleTableScrollTop = ref(0);

......

</script>

<template>
  <!-- 提供数据给子类 -->
  <CommonApplySampleTable
    :apply-sample-list="applySampleTableData"
    :accept-type="applyBasicInfo.acceptType"
    :scroll-top="applySampleTableScrollTop"
    operate-command-type="info-add" />
</template>

子组件 CommonApplySampleTable.vue

javascript 复制代码
<script setup lang="ts" name="CommonApplySampleTable">
import type { ApplySample } from "@/interface";
import ApplySampleSZTable from "@/components/ApplySampleSZTable.vue";
import ApplySampleOtherTable from "@/components/ApplySampleOtherTable.vue";

// 接收父类提供的数据
const props = withDefaults(
  defineProps<{
    // 受理样品列表
    applySampleList: ApplySample[];
    // 受理类别
    acceptType: string;
    // 操作指令类型:新增删除:info-add;修改:info-modify;查看:info-view
    operateCommandType: "info-add" | "info-modify" | "info-view";
    // 滚动到离顶部表头的距离
    scrollTop?: number;
  }>(),
  {
    applySampleList: () => [],
    acceptType: "",
    operateCommandType: "info-add",
    scrollTop: 0
  }
);
</script>

<template>
  <!-- 受理样品表格,水质类 -->
  <ApplySampleSZTable
    v-if="props.acceptType.includes(`GD`)"
    :applySampleData="props.applySampleList"
    :applySampleTableScrollTop="props.scrollTop"
    :operate-command-type="props.operateCommandType" />
  <!-- 受理样品表格,其他 -->
  <ApplySampleOtherTable
    v-else
    :applySampleData="props.applySampleList"
    :applySampleTableScrollTop="props.scrollTop"
    :operate-command-type="props.operateCommandType" />
</template>

<style scoped lang="scss"></style>

祖先传后代模式

父组件 SampleInput.vue

javascript 复制代码
<script setup lang="ts" name="SampleInput">
import { ref, provide, computed } from "vue";
import type { ApplyBasicInfo, Apply, ApplySample } from "@/interface";
import CommonApplySampleTable from "@/components/common/CommonApplySampleTable.vue";

// 定义数据
// 受理基础信息
const applyBasicInfo = ref<ApplyBasicInfo | null>(null)
// 受理样品表格数据
const applySampleTableData = ref<ApplySample[]>([]);
// 滚动到离顶部表头的距离
const applySampleTableScrollTop = ref(0);

......

// 提供数据给后代
// 受理样品列表
provide("applySampleList", applySampleTableData); // applySampleTableData是个Ref对象,不需要.value,传递ref对象本身(响应式数据)
// 受理类别
provide(
  "acceptType",
  computed(() => applyBasicInfo.value.acceptType) // applyBasicInfo.value.acceptType 是派生数据(属性数据),通过computed提供响应式数据
);
// 操作指令类型:新增删除:info-add;修改:info-modify;查看:info-view
provide("operateCommandType", ref("info-add"));
// 滚动到离顶部表头的距离
provide("scrollTop", applySampleTableScrollTop); // applySampleTableScrollTop是个Ref对象,不需要.value,传递ref对象本身(响应式数据)
</script>

<template>
  <CommonApplySampleTable />
</template>

子组件 CommonApplySampleTable.vue

javascript 复制代码
<script setup lang="ts" name="CommonApplySampleTable">
import type { ApplySample } from "@/interface";
import ApplySampleSZTable from "@/components/ApplySampleSZTable.vue";
import ApplySampleOtherTable from "@/components/ApplySampleOtherTable.vue";
import { inject, type Ref, ref } from "vue";

// 接收祖先提供的数据
// 受理样品列表
const applySampleList = inject<Ref<ApplySample[]>>("applySampleList", ref([]));
// 受理类别
const acceptType = inject<Ref<string>>("acceptType", ref(""));
// 操作指令类型:新增删除:info-add;修改:info-modify;查看:info-view
const operateCommandType = inject<Ref<"info-add" | "info-modify" | "info-view">>("operateCommandType", ref("info-add"));
// 滚动到离顶部表头的距离
const scrollTop = inject<Ref<number>>("scrollTop", ref(0));
</script>

<template>
  <!-- 受理样品表格,水质类 -->
  <ApplySampleSZTable
    v-if="acceptType.includes(`GD`)"
    :applySampleData="applySampleList"
    :applySampleTableScrollTop="scrollTop"
    :operate-command-type="operateCommandType" />
  <!-- 受理样品表格,其他 -->
  <ApplySampleOtherTable
    v-else
    :applySampleData="applySampleList"
    :applySampleTableScrollTop="scrollTop"
    :operate-command-type="operateCommandType" />
</template>

<style scoped lang="scss"></style>
相关推荐
白兰地空瓶13 小时前
🚀你以为你在写 React?其实你在“搭一套前端操作系统”
前端·react.js
爱上妖精的尾巴13 小时前
6-4 WPS JS宏 不重复随机取值应用
开发语言·前端·javascript
似水流年QC14 小时前
深入探索 WebHID:Web 标准下的硬件交互实现
前端·交互·webhid
陪我去看海14 小时前
测试 mcp
前端
speedoooo14 小时前
在现有App里嵌入一个AI协作者
前端·ui·小程序·前端框架·web app
全栈胖叔叔-瓜州15 小时前
关于llamasharp 大模型多轮对话,模型对话无法终止,或者输出角色标识User:,或者System等角色标识问题。
前端·人工智能
三七吃山漆15 小时前
攻防世界——wife_wife
前端·javascript·web安全·网络安全·ctf
用户479492835691515 小时前
面试官问"try-catch影响性能吗",我用数据打脸
前端·javascript·面试
GISer_Jing15 小时前
前端营销技术实战:数据+AI实战指南
前端·javascript·人工智能
GIS之路16 小时前
使用命令行工具 ogr2ogr 将 CSV 转换为 Shp 数据(二)
前端