详解Vue学习笔记入门篇之组件的内容分发(slot)


Posted in Javascript onJuly 17, 2017

介绍

为了让组件可以组合,我们需要一种方式来混合父组件的内容与子组件自己的模板。这个过程被称为 内容分发 (或 “transclusion” 如果你熟悉 Angular)。Vue.js 实现了一个内容分发 API,使用特殊的 'slot' 元素作为原始内容的插槽。

编译作用域

在深入内容分发 API 之前,我们先明确内容在哪个作用域里编译。假定模板为:

<child-component>
 {{ message }}
</child-component>

message 应该绑定到父组件的数据,还是绑定到子组件的数据?答案是父组件。组件作用域简单地说是:

  1. 父组件模板的内容在父组件作用域内编译;
  2. 子组件模板的内容在子组件作用域内编译。

一个常见错误是试图在父组件模板内将一个指令绑定到子组件的属性/方法:

<!-- 无效 -->
<child-component v-show="someChildProperty"></child-component>

假定 someChildProperty 是子组件的属性,上例不会如预期那样工作。父组件模板不应该知道子组件的状态。

如果要绑定作用域内的指令到一个组件的根节点,你应当在组件自己的模板上做:

Vue.component('child-component', {
 // 有效,因为是在正确的作用域内
 template: '<div v-show="someChildProperty">Child</div>',
 data: function () {
  return {
   someChildProperty: true
  }
 }
})

类似地,分发内容是在父作用域内编译。

单个slot

除非子组件模板包含至少一个 'slot' 插口,否则父组件的内容将会被丢弃。当子组件模板只有一个没有属性的 slot 时,父组件整个内容片段将插入到 slot 所在的 DOM 位置,并替换掉 slot 标签本身。

最初在 'slot' 标签中的任何内容都被视为备用内容。备用内容在子组件的作用域内编译,并且只有在宿主元素为空,且没有要插入的内容时才显示备用内容。

示例代码:

<div id="app">
  <h1>我是父组件的标题</h1>
  <my-component>
    <p>初始内容1</p>
    <p>初始内容2</p>
  </my-component>
</div>
Vue.component('my-component',{
  template:`
  <div>
    <h2>我是子组件的标题</h2>
    <slot>
      只有在没有要分发的内容是才出现。
    </slot>
  <div>
`,
})
new Vue({
  el:'#app'
})

运行结果如下:

详解Vue学习笔记入门篇之组件的内容分发(slot)

将html部分代码修改为以下代码:

<div id="app">
  <h1>我是父组件的标题</h1>
  <my-component>
  </my-component>
</div>

则运行结果如下:

详解Vue学习笔记入门篇之组件的内容分发(slot)

具名slot

'slot' 元素可以用一个特殊的属性 name 来配置如何分发内容。多个 slot 可以有不同的名字。具名 slot 将匹配内容片段中有对应 slot 特性的元素。

仍然可以有一个匿名 slot,它是默认 slot,作为找不到匹配的内容片段的备用插槽。如果没有默认的 slot,这些找不到匹配的内容片段将被抛弃。

如以下例子:

<div id="app">
  <my-component>
    <h1 slot="header">这是标题</h1>
    <p>第一个段落</p>
    <p>第二个段落</p>
    <p>第三个段落</p>
    <p slot="footer">联系信息</p>
  </my-component>
</div>
Vue.component('my-component',{
  template:`
  <div class="container">
    <header>
      <slot name="header"></slot>
    </header>
    <main>
      <slot></slot>
    </main>
    <footer>
      <slot name="footer"></slot>
    </footer>
  <div>
`,
})
new Vue({
  el:'#app'
})

运行结果如下:

详解Vue学习笔记入门篇之组件的内容分发(slot)

在组合组件时,内容分发 API 是非常有用的机制。

作用域插槽

2.1.0新增

作用域插槽是一种特殊类型的插槽,用作使用一个 (能够传递数据到) 可重用模板替换已渲染元素。

在子组件中,只需将数据传递到插槽,就像你将 props 传递给组件一样。

示例代码:

<div id="app">
  <my-component>
    <template scope="props">
      <p>hello from parent</p>
      <p>{{props.text}}</p>
    </template>
  </my-component>
</div>
Vue.component('my-component',{
  template:`
  <div class="child">
    <slot text="hello from child"></slot>
  <div>
`,
  props:['text']
})
new Vue({
  el:'#app'
})

运行结果:

详解Vue学习笔记入门篇之组件的内容分发(slot)

在父级中,具有特殊属性 scope 的 <'template'> 元素必须存在,表示它是作用域插槽的模板。scope 的值对应一个临时变量名,此变量接收从子组件中传递的 props 对象。

作用域插槽更具代表性的用例是列表组件,允许组件自定义应该如何渲染列表每一项:

<div id="app">
  <my-component :items="items">
    <template slot="item" scope="props">
      <li>{{props.text}}</li>
    </template>
  </my-component>
</div>
Vue.component('my-component',{
  template:`
  <ul>
    <slot name="item" v-for="item in items" :text="item.text"></slot>
  </ul>
`,
  props:['text','items']
})
new Vue({
  el:'#app',
  data:{
    items:[
      {text:'item1'},
      {text:'item2'},
      {text:'item3'},
    ]
  }
})

作用域插槽也可以是具名的

运行结果:

详解Vue学习笔记入门篇之组件的内容分发(slot)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript 开发规范要求(图文并茂)
Jun 11 Javascript
也说JavaScript中String类的replace函数
Sep 22 Javascript
IE6下opacity与JQuery的奇妙结合
Mar 01 Javascript
动态创建script标签实现跨域资源访问的方法介绍
Feb 28 Javascript
一个字符串中出现次数最多的字符 统计这个次数【实现代码】
Apr 29 Javascript
jQuery中的一些小技巧
Jan 18 Javascript
Cookies 和 Session的详解及区别
Apr 21 Javascript
详解Node.js开发中的express-session
May 19 Javascript
node使用Koa2搭建web项目的方法
Oct 17 Javascript
vue2.x集成百度UEditor富文本编辑器的方法
Sep 21 Javascript
IE11下CKEditor在Bootstrap Modal中下拉问题的解决
Sep 25 Javascript
解决Vue的项目使用Element ui 走马灯无法实现的问题
Aug 03 Javascript
详解Vue学习笔记进阶篇之列表过渡及其他
Jul 17 #Javascript
js学使用setTimeout实现轮循动画
Jul 17 #Javascript
详解Vue2.x-directive的学习笔记
Jul 17 #Javascript
javascript  数组排序与对象排序的实例
Jul 17 #Javascript
jQuery常用选择器详解
Jul 17 #jQuery
js轮播图的插件化封装详解
Jul 17 #Javascript
Vue.js中extend选项和delimiters选项的比较
Jul 17 #Javascript
You might like
PHP5在Apache下的两种模式的安装
2006/09/05 PHP
PHP中的socket_read和socket_recv区别详解
2015/02/09 PHP
php微信开发之关注事件
2018/06/14 PHP
一个无限级XML绑定跨框架菜单(For IE)
2007/01/27 Javascript
一个用js实现的页内搜索代码
2007/05/23 Javascript
JS数组去重与取重的示例代码
2014/01/24 Javascript
jQuery+CSS3实现树叶飘落特效
2015/02/01 Javascript
用JavaScript判断CSS浏览器类型前缀的两种方法
2015/10/08 Javascript
JavaScript判断页面加载完之后再执行预定函数的技巧
2016/05/17 Javascript
AngularJS 避繁就简的路由
2016/07/01 Javascript
JavaScript实现Java中Map容器的方法
2016/10/09 Javascript
谈谈Vue.js——vue-resource全攻略
2017/01/16 Javascript
Angularjs 依赖压缩及自定义过滤器写法
2017/02/04 Javascript
JavaScript实现替换字符串中最后一个字符的方法
2017/03/07 Javascript
详解在vue-cli项目下简单使用mockjs模拟数据
2018/10/19 Javascript
封装微信小程序http拦截器过程解析
2019/08/13 Javascript
VUE注册全局组件和局部组件过程解析
2019/10/10 Javascript
[48:02]Ti4循环赛第三日 VG vs Liquid和NEWBEE vs DK
2014/07/12 DOTA
[03:24]2014DOTA2国际邀请赛 神秘商店生意火爆
2014/07/18 DOTA
[49:21]TNC vs VG 2019DOTA2国际邀请赛淘汰赛 胜者组赛BO3 第三场 8.20.mp4
2019/08/22 DOTA
python利用高阶函数实现剪枝函数
2018/03/20 Python
Python键盘输入转换为列表的实例
2018/06/23 Python
Python通用循环的构造方法实例分析
2018/12/19 Python
python加载自定义词典实例
2019/12/06 Python
浅析python 动态库m.so.1.0错误问题
2020/05/09 Python
python自动化测试三部曲之request+django实现接口测试
2020/10/07 Python
波兰品牌鞋履在线商店:Eastend.pl
2020/01/11 全球购物
艺术学院毕业生求职信
2014/07/09 职场文书
竞聘演讲稿开场白
2014/08/25 职场文书
单位授权委托书范本
2014/09/26 职场文书
西安大雁塔导游词
2015/02/10 职场文书
教导处教学工作总结
2015/08/12 职场文书
导游词之山海关
2019/12/10 职场文书
如何使用vue3打造一个物料库
2021/05/08 Vue.js
MySQL实战记录之如何快速定位慢SQL
2022/03/23 MySQL
win10双系统怎么删除一个系统?win10电脑有两个系统删除一个的操作方法
2022/07/15 数码科技