什么是插槽
简单来说就是子组件中的提供给父组件使用的一个坑位,用
<!-- 父组件 -->
<template>
<div>
<Child>Hello Juejin</Child>
</div>
</template>
<script setup lang="ts">
import Child from "./Child.vue";
</script>
<!-- 子组件Child -->
<template>
<div>
<p>1</p>
<slot />
<p>2</p>
</div>
</template>
子组件中的
1
Hello Juejin
2
同样的也可以在标签
<!-- 父组件 -->
<template>
<div>
<Child>{{ msg }}</Child>
</div>
</template>
<script setup lang="ts">
import { ref } from "vue";
import Child from "./Child.vue";
const msg = ref("Hello Juejin");
</script>
先解释一下后面频繁出现的两个词 插槽和插槽内容,防止后面阅读搞混了:
同样的 插槽表示的就是这个 msg 变量。所以子组件 插槽是可以访问到父组件的数据作用域,而插槽内容是无法访问子组件的数据(即父组件中两个
默认内容
在父组件没有提供任何插槽内容的时候,是可以为子组件的插槽指定默认内容的,比如
<!-- 子组件 -->
<template>
<div>
<slot>是默认内容</slot>
</div>
</template>
<!-- 父组件1 -->
<template>
<div>
<Child></Child>
</div>
</template>
<script setup>
import Child from "./Child.vue";
</script>
<!-- 父组件2 -->
<template>
<div>
<Child>Hello Juejin</Child>
</div>
</template>
<script setup>
import Child from "./Child.vue";
</script>
此时父组件 1 展示默认内容
是默认内容
此时父组件 2 展示默认内容
Hello Juejin
具名插槽
很多时候一个 插槽满足不了需求,需要多个插槽。于是就有了具名插槽,就是具有名字的 插槽。简单来说这个具名插槽的目的就是让一个萝卜一个坑,让它们呆在该呆的位置去。比如带 name 的插槽 动态插槽名就是插槽名变成了变量的形式,可以随时修改这个变量从而展示不同的效果。它的写法是 v-slot:[变量名]或者缩写为#[变量名]。 上面说过插槽内容是无法访问子组件的数据的,但是如果想在插槽内容访问子组件的状态该怎么办呢? 其实插槽可以像对组件传递 props 那样,在 slot 标签绑定属性从而传递给父组件中的插槽内容。首先来看下默认插槽的传值方式 还可以以结构的形式获取 slot 提供的数据 注意不能绑定 name 属性,因为绑定了 name 它就成了具名插槽了。同样具名插槽中的 name 属性也不会传递给插槽内容。因为传递的参数只能在插槽内容中使用,所以这类能够接受参数的插槽就被称为了作用域插槽。 下面再看下具名作用域插槽它的传参方式。它接收参数的方式是通过 template 标签的指令 v-slot 的值获取的,所以可以缩写成这样 父组件 子组件<!-- 子组件 -->
<template>
<div>
<!-- 大萝卜 -->
<div>
<slot name="bigTurnip"></slot>
</div>
<!-- 小萝卜 -->
<div>
<slot name="smallTurnip"></slot>
</div>
<!-- 中萝卜 -->
<div>
<slot name="midTurnip"></slot>
</div>
</div>
</template>
<!-- 父组件 -->
<template>
<div>
<Child>
<!-- #smallTurnip 为v-slot:smallTurnip缩写 -->
<template #smallTurnip> 小萝卜 </template>
<template #midTurnip> 中萝卜 </template>
<template #bigTurnip> 大萝卜 </template>
</Child>
</div>
</template>
<script setup>
import Child from "./Child.vue";
</script>
所以父组件中无需在意顺序,只需要写好模板命好名,它就会自动去到它所对应的位置。动态插槽名
<!-- 父组件 -->
<template>
<div>
<Child>
<!-- 等同于#smallTurnip -->
<template #[slotName]> 小萝卜 </template>
<template #midTurnip> 中萝卜 </template>
<template #bigTurnip> 大萝卜 </template>
</Child>
</div>
</template>
<script setup>
import { ref } from "vue";
import Child from "./Child.vue";
const slotName = ref("smallTurnip");
</script>
作用域插槽
作用域插槽
<!-- 子组件 -->
<template>
<div>
<slot personName="xiaoyue" age="18"></slot>
</div>
</template>
<!-- 父组件 -->
<template>
<div>
<Child v-slot="slotProps">
My name is {{ slotProps.personName }} and I am {{ slotProps.age }} years
old this year
</Child>
</div>
</template>
<script setup>
import Child from "./Child.vue";
</script>
<template>
<div>
<Child v-slot="{ personName, age }">
My name is {{ personName }} and I am {{ age }} years old this year
</Child>
</div>
</template>
My name is cat and I am 2 years old this year
具名作用域插槽
<!-- 父组件 -->
<template>
<div>
<Child>
<template #bigTurnip="bigTurnipProps">
{{ bigTurnipProps.message }}
</template>
</Child>
</div>
</template>
<script setup>
import Child from "./Child.vue";
</script>
<!-- 子组件Child.vue -->
<template>
<div>
<!-- 大萝卜 -->
<div>
<slot name="bigTurnip" message="是萝北"></slot>
</div>
</div>
</template>
是萝北
传参
<template #code="{ formState }">
{{ formState }}
</template>
<slot :name="props.slot" :formState="formState"></slot>