Vue 3 setup 中通过 ref 获取子组件实例数据(TS版)

Vue 3.2 版本开始,<script setup> 从实验性质转为稳定版,开发体验十分舒适!

下面的例子都是基于 <script setup> + TS 的情况

总共分三步

1、在子组件暴露需要被父组件调用的属性

2、在子组件定义类型

3、父组件引入子组件的类型,并定义 ref

第一步:defineExpose 暴露子组件属性

defineExpose 官方定义看这里v3.cn.vuejs.org/api/sfc-scr…

使用<script setup> 的组件是默认关闭的,无法被其他组件获得内部的各种属性,比如通过 ref

$parent

如果要被外部访问,需要使用**defineExpose** 暴露需要被外部访问的属性

这里先定义了一个 sonName 属性,然后暴露出去

<script setup lang="ts">
import { ref } from "vue";

const sonName = ref("JackSon");
defineExpose({
  sonName,
});
</script>

第二步:在子组件定义组件实例类型

如果完成第一步的话,在父组件调用子组件实例里某个属性的时候,TS 会报错,找不到该属性(但不影响运行)

所以这里还需要定义一个类型,来声明子组件实例的类型

具体可以看这个 issue

这里定义一个类型 API, 里面是要暴露出去的属性

// Son.vue
<script setup lang="ts">
import { ref } from "vue";
export interface API {
  sonName: string;
}
const sonName = ref("JackSon");
defineExpose({
  sonName,
});
</script>

<template></template>

第三步:父组件引入子组件的类型,并定义 ref

把我们在子组件定义的类型 API 引入父组件,然后在声明子组件的 ref 时使用

看,我们用上了 TS 的提示

// Father.vue
<script setup lang="ts">
import { ref, onMounted } from "vue";
import Son, { API as SonAPI } from "./Son.vue";
const refSon = ref<SonAPI | null>(null);
onMounted(() => {
  if (refSon.value) refSon.value.sonName = "JackSonSon";
});
</script>

<template>
  <Son ref="refSon"></Son>
</template>

题外话

还可以看下 InstanceType ,通过 TS 自带类型,来获取子组件的实例类型

但是对使用 <script setup> 语法糖的组件无效, 可以看下 Vue 官方文档

v3.cn.vuejs.org/guide/types…

子组件使用<script setup>

子组件不使用<script setup>

// Son.vue
<script lang="ts">
import { ref, defineComponent } from "vue";
export default defineComponent({
  name: "Son",
  setup() {
    const sonName = ref("JackSon");
    return {
      sonName,
    };
  },
});
</script>

总结

  • 使用 <script setup> 语法糖的组件,其内部属性是不能被其他组件访问到的
  • 需要使用 defineExpose 将属性暴露出去,并且需要定义一个组件实例的类型,在父组件/其他组件中使用
  • 在不使用<script setup>的情况下,可以用 InstanceType 获取组件实例类型

猜你喜欢

转载自blog.csdn.net/sinat_36728518/article/details/126155877