VUE之组件(插槽slot与可复用组件)

插槽slot

当子组件部分内容是通过父组件传递DOM进行显示时,可以不用父组件props传值的比较挫的语法,Vue中提供了一种新型语法:插槽slot。
废话不多说看代码:

<!-- html代码 -->
		<div id="app">
			<my-blog>
				<h2>格林童话</h2>
				<cite>格林兄弟</cite>
				<p>白雪公主和七个小矮人</p>
			</my-blog>
		</div>
// vue代码
		Vue.component('my-blog',{
			template:`
				<div>
					<slot></slot>
					<span>恶毒的皇后</span>
				</div>`
		})
		var app=new Vue({
			el:"#app",
		})

命名由来:
这种语法看起来像是用子组件直接往里直接插DOM内容,所以称之为插槽。
小结:
1、插槽(Slot)是Vue提出来的一个概念,正如名字一样,插槽用于决定将所携带的内容,插入到指定的某个位置,从而使模板分块,具有模块化的特质和更大的重用性;
2、插槽显不显示、怎样显示是由父组件来控制的,而插槽在哪里显示就由子组件来进行控制

插槽分类有很多种,本节将一一介绍
(1)单个插槽/默认插槽
(2)具名插槽
(3)作用域插槽
(4)解构插槽

插槽分类一:单个插槽(备胎插槽)

出现缘由:
最初在 标签中的任何内容都被视为备用内容。
备用内容在子组件的作用域内编译,并且只有在宿主元素为空,且没有要插入的内容时才显示备用内容。
插槽分类一:单个插槽(备胎)
(1)宿主元素不为空时,显示宿主元素里内容,不显示备用内容

<!-- html代码 -->
		<div id="app">
			<my-blog></my-blog>
		</div>
// vue代码
		Vue.component('my-blog',{
			template:`
				<div>
					<slot>备用内容</slot>
					<span>恶毒的皇后</span>
				</div>`
		})
		var app=new Vue({
			el:"#app",
		})

注意:此时上面没有内容的情况下会自动填上默认的内容
官方文档描述:
Vue 实现了一套内容分发的 API,将 元素作为承载分发内容的出口
插槽内可以包含任何模板代码,包括 HTML模板代码,甚至可以是其它的组件。

通俗理解:
没有插槽的情况下在组件标签内写一些内容是不起任何作用的,当在组件中声明了插槽元素后,在组件元素内写的内容就会跑到它这里了,即插槽此时充当承载分发内容的出口!

具名插槽

首先看个案例,结合案例了解下具名插槽的概念
(1)在子组件中定义了三个slot标签,其中有两个分别添加了name属性header和footer。即通过给slot添加name属性,来指定当前slot的名字

<!-- html代码 -->
		<div id="app">
			<my-blog>
				<h2 slot="header">格林童话</h2>
				<cite>格林兄弟</cite>
				<p slot="footer">白雪公主和七个小矮人</p>
			</my-blog>
		</div>
// vue代码
		Vue.component('my-blog',{
			template:`
				<div>
					<slot name="header"></slot>
					<span>恶毒的皇后</span>
					<slot></slot>
					<slot name="footer"></slot>
				</div>`
		})
		var app=new Vue({
			el:"#app",
		})

具名插槽小结
(1)具名插槽其实就是在父组件中添加一个 slot=‘自定义名字’ 的属性,然后在子组件中的 里面添加 name=‘自定义名字’ 即可
(2)如果父组件中有一部分没有添加 slot 属性,则此处就是默认的插槽,在子组件中的 直接就是使用的父组件的默认插槽部分
(3)如果没有默认插槽,这些找不到匹配的内容片段将被抛弃。

作用域插槽slot

简介:
作用域插槽为Vue2.1.0版本新增,是一种特殊类型的插槽,用作一个 (能被传递数据的) 可重用模板,来代替已经渲染好的元素。
父组件模板的所有东西都会在父级作用域内编译;子组件模板的所有东西都会在子级作用域内编译。
不过,我们可以在父组件中使用 slot-scope 特性从子组件获取数据。前提是需要在子组件中使用 :data=data 先传递 data 的数据。

作用域插槽案例:

<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template slot-scope="props">
					{{props.data}}
				</template>
			</my-blog>
		</div>
	// vue代码 
		// 1.子组件传递数据
		Vue.component('my-blog', {
			template: `<div>
				<slot :data="text"></slot>
				
				</div>`,
			data() {
				return {
					text: '喜欢福安'
				}
			}
		})
		var app = new Vue({
			el: "#app",
		})

注意:
(1)在父级中,具有特殊特性 slot-scope 的 元素必须存在,表示它是作用域插槽的模板(在 2.5.0+,slot-scope 能被用在任意元素或组件中而不再局限于 )。
(2)slot-scope 的值将被用作一个临时变量名,此变量接收从子组件传递过来的 prop 对象
(3)在子组件中,只需将数据传递到插槽,就像你将 prop 传递给组件一样,接下来父组件中使用 slot-scope 特性从子组件获取数据

案例:blog子组件可能在很多地方调用,希望在不同地方调用blog组件时
但是:注意这里要求列表的循环和样式不是由子组件决定,而是外部决定的,修改代码如下

<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template slot-scope="props">
					<h1>{{props.data}}</h1>
				</template>
			</my-blog>
			<my-blog>
				<template slot-scope="props">
					<h4>{{props.data}}</h4>
				</template>
			</my-blog>
		</div>
// vue代码 
		// 1.子组件传递数据
		Vue.component('my-blog', {
			template: `
			<div>
				<ul>
					<li v-for="text of texts">
						<slot :data="text"></slot>
					</li>
				</ul>
			</div>`,
			data() {
				return {
					texts: ["汪汪队","彩虹宝宝","天线宝宝","苏菲亚","大头儿子小头爸爸","熊出没"]
				}
			}
		})
		var app = new Vue({
			el: "#app",
		})

条件判断渲染
到目前为止,便可以在元素上随便操作了
例如:当名字长度等于3的时候,在前面加个“你好”标志
代码如下

<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template slot-scope="props">
					<h1 v-if="props.data.length==3">新推荐--{{props.data}}</h1>
					<h1 v-else>{{props.data}}</h1>
				</template>
			</my-blog>
			
		</div>
	// vue代码 
		// 1.子组件传递数据
		Vue.component('my-blog', {
			template: `
			<div>
				<ul>
					<li v-for="text of texts">
						<slot :data="text"></slot>
					</li>
				</ul>
			</div>`,
			data() {
				return {
					texts: ["汪汪队","彩虹宝宝","天线宝宝","苏菲亚","大头儿子小头爸爸","熊出没"]
				}
			}
		})
		var app = new Vue({
			el: "#app",
		})

作用域插槽slot-scrop新语法
版本:自 2.6.0 起有所更新,已废弃使用 slot-scope 语法,开始使用v-slot进行替代,如下所示

<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template v-slot="props">
					<h1 v-if="props.data.length==3">新推荐--{{props.data}}</h1>
					<h1 v-else>{{props.data}}</h1>
				</template>
			</my-blog>
			
		</div>

解构插槽:
v-slot 的值实际上可以是任何能够作为函数定义中的参数的 JS表达式。所以在支持的环境下 (单文件组件或现代浏览器),你也可以使用 ES2015 解构来传入具体的插槽 prop,如下:

<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template v-slot="{data}">
					<h1 v-if="data.length==3">新推荐--{{data}}</h1>
					<h1 v-else>{{data}}</h1>
				</template>
			</my-blog>
		</div>
// vue代码 
		// 1.子组件传递数据
		Vue.component('my-blog', {
			template: `
			<div>
				<ul>
					<li v-for="text of texts">
						<slot :data="text"></slot>
					</li>
				</ul>
			</div>`,
			data() {
				return {
					texts: ["汪汪队","彩虹宝宝","天线宝宝","苏菲亚","大头儿子小头爸爸","熊出没"]
				}
			}
		})
		var app = new Vue({
			el: "#app",
		})

作用域插槽slot-scrop新语法
你甚至可以定义后备内容,用于插槽 prop 是 undefined 的情形

	<!-- html代码 -->
		<div id="app">
			<!-- 利用solt-scope获取数据 -->
			<my-blog>
				<template v-slot="{data='这部电视禁播了'}">
					<h1 v-if="data.length==3">新推荐--{{data}}</h1>
					<h1 v-else>{{data}}</h1>
				</template>
			</my-blog>
		</div>
// vue代码 
		// 1.子组件传递数据
		Vue.component('my-blog', {
			template: `
			<div>
				<ul>
					<li v-for="text of texts">
						<slot :data="text"></slot>
					</li>
				</ul>
			</div>`,
			data() {
				return {
					texts: ["汪汪队","彩虹宝宝","天线宝宝","苏菲亚",undefined,"熊出没"]
				}
			}
		})
		var app = new Vue({
			el: "#app",
		})

编写可复用组件注意事项:
在编写组件时,最好考虑好以后是否要进行复用。一次性组件间有紧密的耦合没关系,但是可复用组件应当定义一个清晰的公开接口,同时也不要对其使用的外层数据作出任何假设。

Vue 组件的 API 来自三部分——prop、事件和插槽:
1、Prop 允许外部环境传递数据给组件;
2、事件允许从组件内触发外部环境的副作用;
3、插槽允许外部环境将额外的内容组合在组件中。

欢迎来吐槽

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值