Vue 是一个用于构建 Web 应用程序的前端框架,其设计方式使得开发人员可以非常快速地提高工作效率。该框架的各个方面都有很多资料,它的社区也每天都在不断成长。如果你读到了这篇文章,那么这些事儿你很可能已经知道咯。

虽然我们可以快速直接地启动并运行它,但是框架里面那些更复杂和更强大的地方还是需要好好动动脑子才能理解(至少对我是这样)。其中一个是插槽,还有另一个与之相关但功能上不太相同的就是作用域插槽。我学习的时候花了好一阵才理解插槽工作的机制,所以我觉得将我对插槽的理解分享出来是有价值的,因为这没准会帮助到大家。

插槽和具名插槽

父组件以另外一种方式(不是通过常规的 Props 属性传递机制)向子组件传递信息。我发现把这种方法同常规的 HTML 元素联系起来很有帮助。

比如说 HTML 标签。

  1. <a href=”/sometarget">This is a link</a>
  2. 复制代码

如果这是在 Vue 环境中并且<a>是你的组件,那么你需要发送“This is a link”信息到‘a’组件里面,然后它将被渲染成为一个超链接,而“This is a link”就是这个链接的文本。

让我们定义一个子组件来展示它的机制是怎样的:

  1. <template>
  2. <div>
  3. <slot></slot>
  4. </div>
  5. </template>
  6. 复制代码

然后在父组件我们这么做:

  1. <template>
  2. <div>
  3. <child-component>This is from outside</child-component>
  4. </div>
  5. </template>
  6. 复制代码

这时候屏幕上呈现的就应该和你预期的一样就是“This is from outside”,但这是由子组件所渲染出来的。

我们还可以给子组件添加默认的信息,以免到时候这里出现什么都没有传入的情况,就像这样子:

  1. <template>
  2. <div>
  3. <slot>Some default message</slot>
  4. </div>
  5. </template>
  6. 复制代码

然后如果我们像这样子创建我们的子组件:

  1. <child-component>
  2. </child-component>
  3. 复制代码

我们可以看到屏幕上会呈现“Some default message”。

具名插槽和常规插槽非常类似,唯一的差别就是你可以在你的目标组件多个位置传入你的文本。

我们把子组件升级一下,让它有多个具名插槽

  1. <template>
  2. <div>
  3. <slot>Some default message</slot>
  4. <br/>
  5. <slot name="top"></slot>
  6. <br/>
  7. <slot name="bottom"></slot>
  8. </div>
  9. </template>
  10. 复制代码

这样,在我们的子组件中就有了三个插槽。其中 top 和 bottom 插槽是具名插槽。

让我们更新父组件以使用它。

  1. <child-component v-slot:top>
  2. Hello there!
  3. </child-component>
  4. 复制代码

注意 —— 我们在这里使用新的 Vue 2.6 语法来指定我们想要定位的插槽:`v-slot:theName`。

你现在认为会在屏幕上看到什么呢?如果你说是“Hello Top!”,那么你就只说对了一部分。

因为我没有为没有具名的插槽赋予任何值,我们因此也还会得到默认值。所以我们真正会看到的是:

Some default message
Hello There!

其实真正意义上没有具名的插槽是被当作‘default’,所以你还可以这么做:

  1. <child-component v-slot:default>
  2. Hello There!
  3. </child-component>
  4. 复制代码

现在我们就只会看到:

Hello There!

因为我们已经提供了值给默认(也就是未具名)插槽,因此具名插槽‘top’和‘bottom’也都没有默认值。

你发送的并不一定只是文本,还可以是其他组件或者 HTML。你可以发送任意你想展示的内容。

作用域插槽

我认为插槽和具名插槽相对简单,一旦你稍微玩玩就可以掌握。可另一方面,作用域插槽虽然名字相似但又有些不同之处。

我倾向于认为作用域插槽有点像一个放映机(或者是一个我欧洲朋友的投影仪)。以下是原因。

子组件中的作用域插槽可以为父组件中的插槽的显示提供数据。这就像一个人带着放映机站在你的子组件里面,然后在父组件的墙上让一些图像发光。

这有一个例子。在子组件中我们像这样设置了一个插槽:

  1. <template>
  2. <div>
  3. <slot name="top" :myUser="user"></slot>
  4. <br/>
  5. <slot name="bottom"></slot>
  6. <br/>
  7. </div>
  8. </template>
  9. <script>
  10. data() {
  11. return {
  12. user: "Ross"
  13. }
  14. }
  15. </script>
  16. 复制代码

注意到我们的具名插槽‘top’现在有了一个名为‘myUser’的属性,然后我们绑定了一个动态的值在‘user’中。

在我们的父组件中就像这样子设置子组件:

  1. <div>
  2. <child-component v-slot:top="slotProps">{{ slotProps }}</child-component>
  3. </div>
  4. 复制代码

我们在屏幕上看到的就是这样子:

{ “myUser”: “Ross” }

还是使用放映机的类比,我们的子组件通过 myUser 对象将其用户字符串的值传递给父组件。它在父组件上投射到的墙就被称为‘slotProps’。

我知道这不是一个完美的类比,但当我第一次尝试理解这个机制的时候,它帮助我以这种方式思考。

Vue 的文档非常好,而且我也已经看到了一些其他关于作用域插槽工作机制的说明。但很多人采取的方法似乎是将父组件中的所有或部分属性命名为与子组件相同,我认为这会使得数据很难被追踪。

在父组件中使用 ES6 解构,我们这样子写还可以将特定 user 对象从插槽属性(你可以随便怎么称呼它)解脱出来:

  1. <child-component v-slot:top="{myUser}">{{ myUser }}</child-component>
  2. 复制代码

或者甚至就只是在父组件中给它一个新的名字:

  1. <child-component v-slot:top="{myUser: aFancyName}">{{ aFancyName }}</child-component>
  2. 复制代码

所有都是通过 ES6 解构,与 Vue 本身并没有什么关系。

如果你正开始使用 Vue 和插槽,希望这可以让你起步并解决一些棘手的问题。

[译] 我最终是怎么玩转了 Vue 的作用域插槽的更多相关文章

  1. 每日一译系列-模块化css怎么玩(译文)

    原文链接:How Css Modules Work 原文作者是Preact的作者 这是一篇关于如何使用Css Modules的快速介绍,使用到的工具是Webpack吊炸的css-loader 首先,我 ...

  2. [译] 2017 年比较 Angular、React、Vue 三剑客

    原文地址:Angular vs. React vs. Vue: A 2017 comparison 原文作者:Jens Neuhaus 译文出自:掘金翻译计划 本文永久链接:github.com/xi ...

  3. 【半译】在ASP.NET Core中创建内部使用作用域服务的Quartz.NET宿主服务

    在我的上一篇文章中,我展示了如何使用ASP.NET Core创建Quartz.NET托管服务并使用它来按计划运行后台任务.不幸的是,由于Quartz.NET API的工作方式,在Quartz作业中使用 ...

  4. vue邪道玩法 : 把vue实例存在别的地方,以及可能会遇到的问题

    一般来说,VUE项目中,this指向VUE实例. 但有的时候,某些代码会改变this的指向. 这时,可以用一个临时变量存储VUE实例. test1(){ var _this = this // 把vu ...

  5. 一起学习vue源码 - Vue2.x的生命周期(初始化阶段)

    作者:小土豆biubiubiu 博客园:https://www.cnblogs.com/HouJiao/ 掘金:https://juejin.im/user/58c61b4361ff4b005d9e8 ...

  6. 你还不知道Vue的生命周期吗?带你从Vue源码了解Vue2.x的生命周期(初始化阶段)

    作者:小土豆biubiubiu 博客园:https://www.cnblogs.com/HouJiao/ 掘金:https://juejin.im/user/58c61b4361ff4b005d9e8 ...

  7. 适合码农工作时玩的游戏:Scrum

    适合码农工作时玩的游戏:Scrum 昨天遇到一个来自微软的面试者,在面试的最后,我简单介绍了一下我们团队使用一周一次的 Scrum 来做项目管理.他回答说:” 我在微软也用 Scrum,不过我们一周两 ...

  8. [Vue 牛刀小试]:第十七章 - 优化 Vue CLI 3 构建的前端项目模板(1)- 基础项目模板介绍

    一.前言 在上一章中,我们开始通过 Vue CLI 去搭建属于自己的前端 Vue 项目模板,就像我们 .NET 程序员在使用 asp.net core 时一样,我们更多的会在框架基础上按照自己的开发习 ...

  9. Vue基础项目模板

    https://github.com/wanglong/vue-element-admin.git 优化 Vue CLI 3 构建的前端项目模板(1)- 基础项目模板介绍 一站式开源运维平台,分享给大 ...

随机推荐

  1. 整数翻转C++实现 java实现 leetcode系列(七)

    给出一个 32 位的有符号整数,你需要将这个整数中每位上的数字进行反转. 示例 1: 输入: 123 输出: 321 示例 2: 输入: -123 输出: -321 示例 3: 输入: 120 输出: ...

  2. ECharts 在winform中使用(访问JS)

    ECharts 是百度的一个开源chart 数据统计库,采用html5 + js 编程方式. 有比较好的动态效果,功能很强大.能做出酷弦的效果. ECharts 一般用于web 开发.但winform ...

  3. localStorage、sessionStorage的区别

    1.localStorage生命周期是永久的, sessionStorage生命周期为当前窗口或标签页,一旦窗口或标签页被永久关闭了,那么所有通过sessionStorage存储的数据也就被清空了. ...

  4. RabbitMq笔记(2)

    今天收获不少,记个笔记. namespace RabbitMQTest { class Program { static void Main(string[] args) { Consumer(); ...

  5. hiho 1590 - 紧张的会议室。区间问题

    题目链接 小Hi的公司最近员工增长迅速,同时大大小小的会议也越来越多:导致公司内的M间会议室非常紧张. 现在小Hi知道公司目前有N个会议,其中第i个会议的时间区间是(Si, Ei). 注意这里时间区间 ...

  6. swift语言点评六-Numbers and Basic Values

    Topics Logical Values struct Bool A value type whose instances are either true or false. Numeric Val ...

  7. 联想E490 加M.2固态硬盘 卡在第一画面不动解决办法

    电脑配置:  E490  500G机械硬盘,自己加M.2 NVMe 固态硬盘. 问题:启动时出现 2101:Detection error on SSD1(M.2),  无法识别到机械硬盘 (重新插拨 ...

  8. [arc076e]connected?

    题意: 给出一个$R\times C$的棋盘,其中$1$到$N$之间的每个正整数都会在棋盘上出现两次,第$i$个数出现的位置是$(X_{i,1},Y_{i,1})$和$(X_{i,2},Y_{i,2} ...

  9. Linux ping 不通 域名 添加DNS

    修改路由配置文件 vi /etc/resolv.conf # Generated by NetworkManager #NDS nameserver 192.168.32.2 redhat7 系统优化 ...

  10. 洛谷 P1275 魔板

    P1275 魔板 题目描述 有这样一种魔板:它是一个长方形的面板,被划分成n行m列的n*m个方格.每个方格内有一个小灯泡,灯泡的状态有两种(亮或暗).我们可以通过若干操作使魔板从一个状态改变为另一个状 ...