「译」一个案例搞懂 Vue.js 的做用域插槽

做用域插槽是 Vue.js 中一个颇有用的特性,能够显著提升组件的通用性和可复用性。问题在于,它实在不太好理解。尝试搞清楚父子做用域之间错综复杂的关系,其痛苦程度不亚于求解一个棘手的数学方程。html

当你没法理解一个东西的时候,最好的办法就是在解决问题的过程当中体会它的应用。本文将向你展现如何使用做用域插槽构建一个可复用的列表组件。vue

注意: 完整代码能够去 Codepen 查看

最基础的组件

咱们即将构建的组件叫作 my-list ,用来展现一系列的项目。它的特别之处就在于,你能够在每次使用组件的时候自定义列表项目的渲染方式。数组

咱们先从最简单的单个列表开始:一个包含几何图形名字和边数的数组。app

app.jside

Vue.component('my-list', {
  template: '#my-list',
  data() {
    return {
      title: 'Shapes',
      shapes: [ 
        { name: 'Square', sides: 4 }, 
        { name: 'Hexagon', sides: 6 }, 
        { name: 'Triangle', sides: 3 }
      ]
    };
  }
});

new Vue({
  el: '#app'
});

index.html测试

<div id="app">
  <my-list></my-list>
</div>

<script type="text/x-template" id="my-list">
  <div class="my-list">
    <div class="title">{{ title }}</div>
    <div class="list">
      <div class="list-item" v-for="shape in shapes">
        <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>
      </div>
    </div>
  </div>
</script>

在加上一点样式,大概就会是下图这个样子:spa

更通用的 my-list

如今咱们想要让 my-list 更加通用,能够渲染任何类型的列表。此次咱们展现的是一堆颜色的名字以及对应的颜色方块。3d

为此,咱们须要将上例列表独有的数据进行抽象化。因为列表中的项目可能有不一样的结构,咱们将会给 my-list 一个插槽,让父组件来定义列表的展现方式。code

app.jscomponent

Vue.component('my-list', {
  template: '#my-list',
  props: [ 'title' ]
});

index.html

<script type="text/x-template" id="my-list">
  <div class="my-list">
    <div class="title">{{ title }}</div>
    <div class="list">
      <slot></slot>
    </div>
  </div>
</script>

如今,咱们在根实例中建立 my-list 组件的两个实例,分别展现两个测试用例列表:lists:

app.js

new Vue({
  el: '#app',
  data: {
    shapes: [ 
      { name: 'Square', sides: 4 }, 
      { name: 'Hexagon', sides: 6 }, 
      { name: 'Triangle', sides: 3 }
    ],
    colors: [
      { name: 'Yellow', hex: '#F4D03F', },
      { name: 'Green', hex: '#229954' },
      { name: 'Purple', hex: '#9B59B6' }
    ]
  }
});
<div id="app">
  <my-list :title="Shapes">
    <div class="list-item" v-for="item in shapes">
      <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>
    </div>
  </my-list>
  <my-list :title="Colors">
    <div class="list-item" v-for="color in colors">
      <div>
        <div class="swatch" :style="{ background: color.hex }"></div>
        {{ color.name }}
      </div>
    </div>
  </my-list>
</div>

效果以下图:

大材小用的组件

咱们刚才建立的组件确实符合要求,但那段代码算不上很好。my-list 原本应该是一个展现列表的组件,但咱们却把渲染列表须要的逻辑部分抽象到了父组件中,这样一来,子组件在这里只不过是用来包裹列表而已,未免显得大材小用了。

更糟糕的是,在两个组件的声明中存在着大量重复代码(例如,<div class="list-item" v-for="item in ...">)。若是咱们可以在子组件中编写这些代码,那么子组件就再也不是“打酱油的角色”了。

做用域插槽

普通插槽没法知足咱们的需求,这时候,做用域插槽就派上用场了。做用域插槽容许你传递一个模板而不是已经渲染好的元素给插槽。之因此叫作”做用域“插槽,是由于模板虽然是在父级做用域中渲染的,却能拿到子组件的数据。

例如,带有做用域插槽的组件 child 大概是下面这个样子:

<div>
  <slot my-prop="Hello from child"></slot>
</div>

使用这个组件的父组件将会在插槽中声明一个 template 元素。这个模板元素会有一个 scope (译者注:Vue 2.6 后改成 v-slot 属性)属性指向一个对象,任何添加到插槽(位于子组件模板)中的属性都会做为这个对象的属性。

<child>
  <template scope="props">
    <span>Hello from parent</span>
    <span>{{ props.my-prop }}</span>
  </template>
</child>

将会渲染成:

<div>
  <span>Hello from parent</span>
  <span>Hello from child</span>
</div>

my-list 中使用做用域插槽

咱们将两个列表数组经过 props 传递给 my-list。以后将普通插槽替换为做用域插槽,这样,my-list 就可以负责迭代列表项目,同时父组件依然可以定义每一个项目具体的展现方式。

index.html

<div id="app">
  <my-list title="Shapes" :items="shapes">
    <!--在这里书写 template-->
  </my-list>
  <my-list title="Colors" :items="colors">
    <!--在这里书写 template-->
  </my-list>   
</div>

接着咱们让 my-list 迭代项目。在 v-for 循环中,item 是当前迭代项目的别名。咱们能够建立一个插槽并经过 v-bind="item" 将那个项目绑定到插槽中。

app.js

Vue.component('my-list', {
  template: '#my-list',
  props: [ 'title', 'items' ]
});

index.html

<script type="text/x-template" id="my-list">
  <div class="my-list">
    <div class="title">{{ title }}</div>
    <div class="list">
      <div v-for="item in items">
        <slot v-bind="item"></slot>
      </div>
    </div>
  </div>
</script>
注意:也许你以前没见过不带参数的 v-bind 用法。这种用法将会把整个对象的因此属性都绑定到当前元素上。在涉及做用域插槽时,这种用法很常见,由于绑定的对象可能有不少属性,而一一将它们列举出来并手动绑定显然太麻烦了。

如今,回到根实例这里来,在 my-list 的插槽中声明一个模板。首先看一下几何图形列表(第一个例子中的列表),咱们声明的模板必须带有一个 scope 属性,这里将其赋值为 shapeshape 这个别名可让咱们访问做用域插槽。在模板中,咱们能够继续沿用最初例子中的标记来展现项目。

<my-list title="Shapes" :items="shapes">
  <template scope="shape">
    <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>
  </template>
</my-list>

整个模板大概是下面这样:

<div id="app">
  <my-list title="Shapes" :items="shapes">
    <template scope="shape">
      <div>{{ shape.name }} <small>({{ shape.sides }} sides)</small></div>
    </template>
  </my-list>
  <my-list title="Colors" :items="colors">
    <template scope="color">
      <div>
        <div class="swatch" :style="{ background: color.hex }"></div>
        {{ color.name }}
      </div>
    </template>
  </my-list>   
</div>

结论

虽然用上做用域插槽以后,代码量并未减小,可是咱们将通用的功能都交由子组件负责,这显著提升了代码的健壮性。

完整代码的 Codepen 在这里:

https://codepen.io/anthonygor...

译者注: Vue.js 2.6.0 以后将 slot-scope 改成 v-slot
相关文章
相关标签/搜索