Vue <slot> 元素

实例

使用内置的 <slot> 元素将父组件的内容放置在子组件的 <template> 中。

  1. <template>
  2. <div>
  3. <p>SlotComp.vue</p>
  4. <slot></slot>
  5. </div>
  6. </template>

定义与用法

内置的 <slot> 元素用于放置从父组件接收的内容。

当调用子组件时,在开始标记和结束标记之间提供的内容将终止于 <slot> 元素位于该子组件内的位置。

一个组件可以容纳多个 <slot>,并且这些 <slot> 可以用名称 prop 命名。对于具有不同命名 slot 的此类组件,我们可以使用 v-slot 指令将内容发送到特定 slot。(实例3)如果父级未提供任何内容,则 <slot> 元素的开始标记和结束标记之间的内容将用作回退内容。(实例5)信息可以通过 <slot> prop 提供给父元素。(实例8)<slot> 元素只是内容的占位符,<slot> 元素本身并没有呈现为 DOM 元素。


Props

Prop描述
[any]slot 定义的 prop 会创建 'scoped slots' 范围插槽,并且这些 prop 会发送到父级。
name命名一个 slot,以便父级可以使用 v-slot 指令将内容引导到特定的 slot 中。

更多实例

实例 1

使用 slot 来包装较大的动态 HTML 内容块,以获得类似卡片的外观。

App.vue:

  1. <template>
  2. <h3>Slots in Vue</h3>
  3. <p>We create card-like div boxes from the foods array.</p>
  4. <div id="wrapper">
  5. <slot-comp v-for="x in foods">
  6. <img v-bind:src="x.url">
  7. <h4>{{x.name}}</h4>
  8. <p>{{x.desc}}</p>
  9. </slot-comp>
  10. </div>
  11. </template>

当内容进入 <slot> 所在的组件时,我们在 <slot> 周围使用 div,并在本地设置 <div> 的样式,以在不影响应用程序中其他 div 的情况下,在内容周围创建类似卡片的外观。

SlotComp.vue:

  1. <template>
  2. <div> <!-- This div makes the card-like appearance -->
  3. <slot></slot>
  4. </div>
  5. </template>
  6. <script></script>
  7. <style scoped>
  8. div {
  9. box-shadow: 0 4px 8px 0 rgba(0,0,0,0.2);
  10. border-radius: 10px;
  11. margin: 10px;
  12. }
  13. </style>
实例 2

使用 slot 创建页脚。

App.vue:

  1. <template>
  2. <h3>Reusable Slot Cards</h3>
  3. <p>We create card-like div boxes from the foods array.</p>
  4. <p>We also create a card-like footer by reusing the same component.</p>
  5. <div id="wrapper">
  6. <slot-comp v-for="x in foods">
  7. <img v-bind:src="x.url">
  8. <h4>{{x.name}}</h4>
  9. </slot-comp>
  10. </div>
  11. <footer>
  12. <slot-comp>
  13. <h4>Footer</h4>
  14. </slot-comp>
  15. </footer>
  16. </template>
实例 3

使用 slot 名称,可以将内容发送到特定 slot。

SlotComp.vue:

  1. <h3>Component</h3>
  2. <div>
  3. <slot name="topSlot"></slot>
  4. </div>
  5. <div>
  6. <slot name="bottomSlot"></slot>
  7. </div>

App.vue:

  1. <h1>App.vue</h1>
  2. <p>The component has two div tags with one slot in each.</p>
  3. <slot-comp v-slot:bottomSlot>'Hello!'</slot-comp>
实例 4

在一个组件中有两个 slot 的情况下,发送到该组件的内容将同时出现在这两个 slot 中。

App.vue:

  1. <h1>App.vue</h1>
  2. <p>The component has two div tags with one slot in each.</p>
  3. <slot-comp>'Hello!'</slot-comp>

SlotComp.vue:

  1. <h3>Component</h3>
  2. <div>
  3. <slot></slot>
  4. </div>
  5. <div>
  6. <slot></slot>
  7. </div>
实例 5

在 slot 中使用回退内容,以便在父级未提供内容时呈现某些内容。

App.vue:

  1. <template>
  2. <h3>Slots Fallback Content</h3>
  3. <p>A component without content provided can have fallback content in the slot tag.</p>
  4. <slot-comp>
  5. <!-- Empty -->
  6. </slot-comp>
  7. <slot-comp>
  8. <h4>This content is provided from App.vue</h4>
  9. </slot-comp>
  10. </template>

SlotComp.vue:

  1. <template>
  2. <div>
  3. <slot>
  4. <h4>This is fallback content</h4>
  5. </slot>
  6. </div>
  7. </template>
实例 6

没有名称的 slot 将是父级内容的默认 slot。

SlotComp.vue:

  1. <h3>Component</h3>
  2. <div>
  3. <slot></slot>
  4. </div>
  5. <div>
  6. <slot name="bottomSlot"></slot>
  7. </div>

App.vue:

  1. <h1>App.vue</h1>
  2. <p>The component has two div tags with one slot in each.</p>
  3. <slot-comp>'Hello!'</slot-comp>
实例 7

使用 v-slot:default 以明确地将 slot 定义为默认 slot。

SlotComp.vue:

  1. <h3>Component</h3>
  2. <div>
  3. <slot></slot>
  4. </div>
  5. <div>
  6. <slot name="bottomSlot"></slot>
  7. </div>

App.vue:

  1. <h1>App.vue</h1>
  2. <p>The component has two div tags with one slot in each.</p>
  3. <slot-comp v-slot:default>'Default slot'</slot-comp>
实例 8

范围 slot: 使用 slot 中的 'foodName' prop 将食物名称传达给父级。

SlotComp.vue:

  1. <template>
  2. <slot
  3. v-for="x in foods"
  4. :key="x"
  5. :foodName="x"
  6. ></slot>
  7. </template>
  8. <script>
  9. export default {
  10. data() {
  11. return {
  12. foods: ['Apple','Pizza','Rice','Fish','Cake']
  13. }
  14. }
  15. }
  16. </script>

App.vue:

  1. <slot-comp v-slot="food">
  2. <h2>{{ food.foodName }}</h2>
  3. </slot-comp>
实例 9

范围 slot:使用 slot 中的 prop,基于带有对象的数组,将一些东西传达给父对象。

SlotComp.vue:

  1. <template>
  2. <slot
  3. v-for="x in foods"
  4. :key="x.name"
  5. :foodName="x.name"
  6. :foodDesc="x.desc"
  7. :foodUrl="x.url"
  8. ></slot>
  9. </template>
  10. <script>
  11. export default {
  12. data() {
  13. return {
  14. foods: [
  15. { name: 'Apple', desc: 'Apples are a type of fruit that grow on trees.', url: 'img_apple.svg' },
  16. { name: 'Pizza', desc: 'Pizza has a bread base with tomato sauce, cheese, and toppings on top.', url: 'img_pizza.svg' },
  17. { name: 'Rice', desc: 'Rice is a type of grain that people like to eat.', url: 'img_rice.svg' },
  18. { name: 'Fish', desc: 'Fish is an animal that lives in water.', url: 'img_fish.svg' },
  19. { name: 'Cake', desc: 'Cake is something sweet that tastes good but is not considered healthy.', url: 'img_cake.svg' }
  20. ]
  21. }
  22. }
  23. }
  24. </script>

App.vue:

  1. <slot-comp v-slot="food">
  2. <hr>
  3. <h2>{{ food.foodName }}<img :src=food.foodUrl></h2>
  4. <p>{{ food.foodDesc }}</p>
  5. </slot-comp>
实例 10

使用命名范围的 slot 将一个文本放入 "leftSlot",另一个文本则放入 "rightSlot"。

SlotComp.vue:

  1. <template>
  2. <slot
  3. name="leftSlot"
  4. :text="leftText"
  5. ></slot>
  6. <slot
  7. name="rightSlot"
  8. :text="rightText"
  9. ></slot>
  10. </template>
  11. <script>
  12. export default {
  13. data() {
  14. return {
  15. leftText: 'This text belongs to the LEFT slot.',
  16. rightText: 'This text belongs to the RIGHT slot.'
  17. }
  18. }
  19. }
  20. </script>

App.vue:

  1. <slot-comp #leftSlot="leftProps">
  2. <div>{{ leftProps.text }}</div>
  3. </slot-comp>
  4. <slot-comp #rightSlot="rightProps">
  5. <div>{{ rightProps.text }}</div>
  6. </slot-comp>

分类导航