当使用 setup
的时候,组件直接引入就可以了,不需要再自己手动注册
ref
一般用于基本的数据类型,比如 string
,boolean
,reactive
一般用于对象 ref 的地方其实也是调用的 reactive
实现的。
{{ title }}
{{ data }}
// 第一种不带默认值props
const props = defineProps<{
foo: string
bar?: number
}>()
// 第二种带默认值props
export interface ChildProps {
foo: string
bar?: number
}
const props = withDefaults(defineProps(), {
foo: "1qsd"
bar?: 3
})
// 第一种获取事件
const emit = defineEmits<{
(e: 'change', id: number): void
(e: 'update', value: string): void
}>()
// 第二种获取事件
const emit = defineEmits(["dosth"])
useAttrs
可以获取父组件传过来的 id
、class
等值。useSlots
可以获得插槽的内容。例子中,我们使用 useAttrs
获取父组件传过来的 id
、class
、useSlots
获取插槽的内容。
父组件:
{{ fatherRef }}
1223
子组件:
11
在 setup
里边自定义指令的时候,只需要遵循vNameOfDirective
这样的命名规范就可以了
比如如下自定义 focus
指令,命名就是 vMyFocus
,使用的就是 v-my-focus
自定义指令
子组件
父组件
222
父组件
子组件
2222
当从父组件向子组件传 props
的时候,必须使用 toRefs
或者 toRef
进行转一下,这是为什么呢?
这里是因为如果不使用 toRefs
转一次的话,当父组件中的 props
改变的时候,子组件如果使用了 Es6 的解析,会失去响应性。
可以看下如下例子
父组件
{{ fatherRef }}
子组件
{{ fatherRef }}
可以看到当父组件如果点击之后,因为使用 const { fatherRef } = props;
进行解析,就失去了响应性
所以当父组件变成 2 的时候,子组件还是 1。
这里有两种解决办法
使用 const { fatherRef } = toRefs(props)
;
在模版中中使用 props.fatherRef
父组件
{{ fatherRef }}
子组件
例子中父组件传递 modelValue
和 update:modelValue
方法 父组件:
子组件:
组件本身是可以调用组件自身的,也就是递归。vue3 中使用文件名称自动注册为组件的名称,比如名为 Child.vue
的组件可以在其模板中用
引用它自己。这里需要注意的是需要设置条件语句,用来中断递归,不然递归会无限递归下去。
父组件
子组件
2"
>
通过ref直接拿到dom引用
通过对div元素添加了ref属性,为了获取到这个元素,我们声明了一个与ref属性名称相同的变量sectionRef,然后我们通过 sectionRef.value 的形式即可获取该div元素
通过父容器的ref遍历拿到dom引用
{{item}}
通过对父元素添加了ref属性,并声明了一个与ref属性名称相同的变量listRef,此时通过listRef.value会获得包含子元素的dom对象 此时可以通过listRef.value.children[index]
的形式获取子元素dom
通过:ref将dom引用放到数组中
{{item}}
通过:ref循环调用setRefAction
方法,该方法会默认接收一个el参数,这个参数就是我们需要获取的div元素 此时可以通过state.refList[index]
的形式获取子元素dom
通过子组件emit传递ref
{{item}}
通过对子组件添加了ref属性,并声明了一个与ref属性名称相同的变量cellRef,此时可以通过emit将cellRef.value作为一个dom引用传递出去
tsx 等 render 组件中获取的方式更简单
import { defineComponent, ref, onMounted } from "@vue/runtime-core";
import { ElForm } from "element-plus";
export default defineComponent({
setup() {
const $form = ref>(null);
onMounted(() => {
$form.value?.validate; // 类型正确
});
return () => ;
},
});
需要注意的是,如果使用 expose 暴露方法出去,无法获取到对应的类型,您需要自定义类型 github.com/vuejs/rfcs/…[1]
// 组件 MyForm
import { defineComponent, ref, onMounted } from "@vue/runtime-core";
import { ElForm } from "element-plus";
type ELEForm = InstanceType;
// 在外界通过 ref 获取组件实例 请使用这个类型
export interface MyFormExpose {
validate: ELEForm["validate"];
}
export default defineComponent({
name: "MyForm",
setup(props, { expose }) {
const $form = ref>(null);
expose({
validate: (callback) => $form.value?.validate(callback),
} as MyFormExpose);
return () => ;
},
});