我们之前在学习组件的时候知道可以通过props来传递属性:

  1. <template>
  2. <MyButton text="发送请求" />
  3. </template>
  1. <template>
  2. <button>{{ text }}</button>
  3. </template>
  4. <script>
  5. export default{
  6. props: ["text"]
  7. }
  8. </script>

插槽基础

那么如何给子传递传递更丰富的信息呢?例如给子组件传递一张图片,一段 HTML…
Vue 给我们提供了<slot></slot>组件用于在子组件内部进行占位。

  1. <template>
  2. <MyButton>发送请求</MyButton>
  3. </template>
  1. <template>
  2. <button>
  3. <slot></slot>
  4. </button>
  5. </template>
  1. <!-- 最后的渲染结果 -->
  2. <button>发送请求</button>

:::info <slot></slot>是组件化中一个比较形象的占位,相当于内容占位标签,组件化内部的一种扩展功能,表示了父元素提供的插槽内容将在哪里被渲染。
<slot></slot>可以是任意的内容:一段文本、一个组件、一段 HTML 都是可以的,我们只要理解<slot></slot>就是在子组件内的占位标签就可以了。 :::

如果子组件内部没有提供<slot></slot>进行占位,那么你在调用组件的时候在标签中间写的东西都将无效:

  1. <template>
  2. <MyButton>发送请求</MyButton>
  3. </template>
  1. <template>
  2. <!-- 不提供 slot -->
  3. <button></button>
  4. </template>
  1. <!-- 最后的渲染结果 -->
  2. <button></button>

使用插槽的时候可以访问当前组件中的内容,但是不能访问子组件的内容:

  1. <template>
  2. <MyButton>
  3. <span>{{ btnText }}</span>
  4. </MyButton>
  5. </template>
  6. <script>
  7. export default{
  8. data(){
  9. return{
  10. btnText: "发送请求"
  11. }
  12. }
  13. }
  14. </script>
  1. <template>
  2. <!-- 🙅 错误!!! -->
  3. <button>{{ btnText }}</button>
  4. </template>

理解插槽的另一种方式是和下面的 JavaScript 函数作类比,其概念是类似的:

  1. // 父元素传入插槽内容
  2. FancyButton('Click me!')
  3. // FancyButton 在自己的模板中渲染插槽内容
  4. function FancyButton(slotContent) {
  5. return `
  6. <button class="fancy-btn">
  7. ${slotContent}
  8. </button>`
  9. }

我们还可以给插槽设置默认的内容,当父组件调用的时候子组件内不传递任何的内容将会使用默认的内容:

  1. <template>
  2. <MyButton></MyButton>
  3. </template>
  1. <template>
  2. <button>
  3. <slot>默认文本</slot>
  4. </button>
  5. </template>
  1. <!-- 最后的渲染结果 -->
  2. <button>默认文本</button>

具名插槽

理解具名的意思:对于函数来说有两大分类,分别是「匿名函数」和「具名」函数。

  1. // 匿名函数赋值给 test
  2. var test = function () {};
  3. // 将具名函数 test2 赋值给 test1
  4. var test1 = function test2() {};

有没有办法我又要使用template而且slot也可以生效呢?
当你并没有给<slot></slot>设置name名称的时候,slot会产生一个隐式的名称:default

  1. <MyButton>
  2. <template v-slot:default>
  3. <span>提交</span>
  4. </template>
  5. </MyButton>

你也可以给子组件的slot组件加上随意的名称,例如我有一个布局组件,我给每一个slot都加上了name属性:

  1. <div class="container">
  2. <header>
  3. <slot name="header">默认头部信息</slot>
  4. </header>
  5. <main>
  6. <!-- 该 slot 会默认产生一个名为 default 的名称 -->
  7. <slot>默认主体信息</slot>
  8. </main>
  9. <footer>
  10. <slot name="footer">默认底部信息</slot>
  11. </footer>
  12. </div>

而我们在父组件使用该组件的时候就可以根据name来选择传递的内容。
要为具名插槽传入内容,我们需要使用一个含**v-slot**指令的**<template>**元素,并将目标插槽的名字传给该指令:

  1. <Layout>
  2. <template v-slot:header>
  3. <img src="/logo.png" />
  4. </template>
  5. </Layout>

v-slot也可以简写为#:

  1. <Layout>
  2. <template v-slot:header>
  3. <img src="/logo.png" />
  4. </template>
  5. <!-- 使用默认的 slot , 名字为 default -->
  6. <template #default>
  7. <p>A paragraph for the main content.</p>
  8. </template>
  9. <template #footer>
  10. <p>This is page footer content.</p>
  11. </template>
  12. </Layout>

当一个组件同时接收默认插槽和具名插槽时,所有位于顶级的非**<template>**节点都被隐式地视为默认插槽的内容。所以上面也可以写成:

  1. <Layout>
  2. <template v-slot:header>
  3. <img src="/logo.png" />
  4. </template>
  5. <!-- 隐式的默认插槽 -->
  6. <p>A paragraph for the main content.</p>
  7. <template #footer>
  8. <p>This is page footer content.</p>
  9. </template>
  10. </Layout>

最终的渲染结果就是这样的:

  1. <div class="container">
  2. <header>
  3. <img src="/logo.png" />
  4. </header>
  5. <main>
  6. <p>A paragraph for the main content.</p>
  7. </main>
  8. <footer>
  9. <p>This is page footer content.</p>
  10. </footer>
  11. </div>

作用域插槽

在某些场景下插槽的内容可能想要同时使用父组件域内和子组件域内的数据。要做到这一点,我们需要一种方法来让子组件在渲染时将一部分数据提供给插槽。
我们可以给<slot></slot>传递属性供父组件进行使用:

  1. <div>
  2. <!-- 和普通组件一样,在标签上传递属性 -->
  3. <slot :text="greetingMessage" :count="1"></slot>
  4. </div>

父组件可以来接收<slot></slot>传递过来的值:

  1. <MyComponent v-slot="props">
  2. {{ props }}
  3. <!-- { 'text':"xxx", 'count': 'xxx' } -->
  4. </MyComponent>

子组件通过插槽传递来的数据会包装到**props**对象上!!!

如果子组件内存在多个具名插槽,我们也可以进行搭配使用:

  1. <MyComponent>
  2. <!-- #header="headerProps" 是 v-slot:header="headerProps" 的简写形式 -->
  3. <template #header="headerProps">
  4. {{ headerProps }}
  5. </template>
  6. <template #default="defaultProps">
  7. {{ defaultProps }}
  8. </template>
  9. <template #footer="footerProps">
  10. {{ footerProps }}
  11. </template>
  12. </MyComponent>

:::warning ⚠️ 注意
注意插槽上的name是一个 Vue 特别保留的 attribute,不会作为 props 传递给插槽。因此最终headerProps的结果将不包含name属性!!! :::

如果你混用了具名插槽与默认插槽,则需要为默认插槽使用显式的**<template>**标签。尝试直接为组件添加**v-slot**指令将导致编译错误:

  1. <!-- 该模板无法编译 -->
  2. <template>
  3. <MyComponent v-slot="{ message }">
  4. <p>{{ message }}</p>
  5. <template #footer>
  6. <!-- message 属于默认插槽,此处不可用 -->
  7. <p>{{ message }}</p>
  8. </template>
  9. </MyComponent>
  10. </template>