目标: 多个组件使用同一个挂载点,并动态切换,这就是动态组件
需求: 完成一个注册功能页面, 2个按钮切换, 一个填写注册信息, 一个填写用户简介信息
效果如下:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pWe03eJv-1625832204058)(images/动态组件.gif)]
准备被切换的 - UserName.vue / UserInfo.vue 2个组件
引入到UseDynamic.vue注册
准备变量来承载要显示的"组件名"
设置挂载点, 使用is属性来设置要显示哪个组件
点击按钮 – 修改comName变量里的"组件名"
下面显示注册组件-动态切换:
在App.vue - 引入01_UseDynamic.vue并使用显示
总结: vue内置component组件, 配合is属性, 设置要显示的组件名字
目标: 组件切换会导致组件被频繁销毁和重新创建, 性能不高
使用Vue内置的keep-alive组件, 可以让包裹的组件保存在内存中不被销毁
演示1: 可以先给UserName.vue和UserInfo.vue 注册created和destroyed生命周期事件, 观察创建和销毁过程
演示2: 使用keep-alive内置的vue组件, 让动态组件缓存而不是销毁
语法:
Vue内置的keep-alive组件 包起来要频繁切换的组件
02_UseDynamic.vue
补充生命周期:
总结: keep-alive可以提高组件的性能, 内部包裹的标签不会被销毁和重新创建, 触发激活和非激活的生命周期方法
目标: 被缓存的组件不再创建和销毁, 而是激活和非激活
补充2个钩子方法名:
activated – 激活时触发
deactivated – 失去激活状态触发
目标: 用于实现组件的内容分发, 通过 slot 标签, 可以接收到写在组件标签内的内容
vue提供组件插槽能力, 允许开发者在封装组件时,把不确定的部分定义为插槽
插槽例子:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Kg1MfoDv-1625832204061)(images/1ddad96d-f925-452b-8c40-85288fc2cbc4.gif)]
需求: 以前折叠面板案例, 想要实现不同内容显示, 我们把折叠面板里的Pannel组件, 添加组件插槽方式
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hHrQkQfG-1625832204064)(images/image-20210307110014556.png)]
语法口诀:
03/Pannel.vue - 组件(直接复制)
芙蓉楼送辛渐
{{ isShow ? "收起" : "展开" }}
寒雨连江夜入吴,
平明送客楚山孤。
洛阳亲友如相问,
一片冰心在玉壶。
views/03_UserSlot.vue - 使用组件(直接复制)
框: 在这个基础重复使用组件
案例:折叠面板
views/03_UseSlot.vue - 组件插槽使用
案例:折叠面板
我是内容
寒雨连江夜入吴,
平明送客楚山孤。
洛阳亲友如相问,
一片冰心在玉壶。
总结: 组件内容分发技术, slot占位, 使用组件时传入替换slot位置的标签
目标: 如果外面不给传, 想给个默认显示内容
口诀: 夹着内容默认显示内容, 如果不给插槽slot传东西, 则使用夹着的内容在原地显示
默认内容
目标: 当一个组件内有2处以上需要外部传入标签的地方
传入的标签可以分别派发给不同的slot位置
要求: v-slot一般用跟template标签使用 (template是html5新出标签内容模板元素, 不会渲染到页面上, 一般被vue解析内部标签)
components/04/Pannel.vue - 留下具名slot
{{ isShow ? "收起" : "展开" }}
views/04_UseSlot.vue使用
案例:折叠面板
芙蓉楼送辛渐
我是内容
我是标题
寒雨连江夜入吴,
平明送客楚山孤。
洛阳亲友如相问,
一片冰心在玉壶。
v-slot可以简化成#使用
v-bind可以省略成: v-on: 可以省略成@ 那么v-slot: 可以简化成#
总结: slot的name属性起插槽名, 使用组件时, template配合#插槽名传入具体标签
目标: 子组件里值, 在给插槽赋值时在父组件环境下使用
复习: 插槽内slot中显示默认内容
例子: 默认内容在子组件中, 但是父亲在给插槽传值, 想要改变插槽显示的默认内容
口诀:
components/05/Pannel.vue - 定义组件, 和具名插槽, 给slot绑定属性和值
芙蓉楼送辛渐
{{ isShow ? "收起" : "展开" }}
{{ defaultObj.defaultOne }}
views/05_UseSlot.vue
案例:折叠面板
{{ scope.row.defaultTwo }}
总结: 组件内变量绑定在slot上, 然后使用组件v-slot=“变量” 变量上就会绑定slot身上属性和值
目标: 了解作用域插槽使用场景, 自定义组件内标签+内容
案例: 封装一个表格组件, 在表格组件内循环产生单元格
准备MyTable.vue组件 – 内置表格, 传入数组循环铺设页面, 把对象每个内容显示在单元格里
准备UseTable.vue – 准备数据传入给MyTable.vue使用
components/06/MyTable.vue - 模板(直接复制)
序号
姓名
年龄
头像
views/06_UseTable.vue - 准备数据, 传入给MyTable.vue组件里循环使用
list: [
{
name: "小传同学",
age: 18,
headImgUrl:
"http://yun.itheima.com/Upload/./Images/20210303/603f2d2153241.jpg",
},
{
name: "小黑同学",
age: 25,
headImgUrl:
"http://yun.itheima.com/Upload/./Images/20210304/6040b101a18ef.jpg",
},
{
name: "智慧同学",
age: 21,
headImgUrl:
"http://yun.itheima.com/Upload/./Images/20210302/603e0142e535f.jpg",
},
],
例子: 我想要给td内显示图片, 需要传入自定义的img标签
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GDyLtlVL-1625832204066)(images/image-20210511170436114.png)]
正确做法:
在MyTable.vue的td中准备占位, 但是外面需要把图片地址赋予给src属性,所以在slot上把obj数据绑定
components/06/MyTable.vue - 正确代码
序号
姓名
年龄
头像
{{ index + 1 }}
{{ obj.name }}
{{ obj.age }}
{{ obj.headImgUrl}}
在UseTable使用MyTable的时候, template上v-slot绑定变量, 传入img组件设置图片地址
总结: 插槽可以自定义标签, 作用域插槽可以把组件内的值取出来自定义内容
自定义指令文档
除了核心功能默认内置的指令 (v-model
和 v-show
),Vue 也允许注册自定义指令。 v-xxx
html+css的复用的主要形式是组件
你需要对普通 DOM 元素进行底层操作,这时候就会用到自定义指令
目标: 获取标签, 扩展额外的功能
局部注册和使用
07_UseDirective.vue - 只能在当前组件.vue文件中使用
全局注册
在main.js用 Vue.directive()方法来进行注册, 以后随便哪个.vue文件里都可以直接用v-fofo指令
// 全局指令 - 到处"直接"使用
Vue.directive("gfocus", {
inserted(el) {
el.focus() // 触发标签的事件方法
}
})
总结: 全局注册自定义指令, 哪里都能用, 局部注册, 只能在当前vue文件里用
目标: 使用自定义指令, 传入一个值
需求: 定义color指令-传入一个颜色, 给标签设置文字颜色
main.js定义处修改一下
// 目标: 自定义指令传值
Vue.directive('color', {
inserted(el, binding) {
el.style.color = binding.value
},
update(el, binding) {
el.style.color = binding.value
}
})
Direct.vue处更改一下
修改文字颜色
总结: v-xxx, 自定义指令, 获取原生DOM, 自定义操作
完成如下案例和各步功能
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-r0xlGXiH-1625832204068)(images/21_tabbar案例_所有效果.gif)]
知识点:
目标: 创建项目文件夹, 引入字体图标, 下载bootstrap, less, [email protected] axios, 在App.vue注册组件
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gXyajoDv-1625832204071)(images/image-20210511172408766.png)]
组件分析:
组件拆分:
三个页面
思路分析:
①: vue create tabbar-demo
②: yarn add less [email protected] -D
③: yarn add bootstrap axios 并在main.js 引入和全局属性
④: 根据需求-创建需要的页面组件
⑤: 把昨天购物车案例-封装的MyHeader.vue文件复制过来复用
⑥: 从App.vue – 引入组织相关标签
新建工程:
vue create tabbar-demo
yarn add less [email protected] -D
yarn add bootstrap axios
在main.js中引入bootStrap.css和字体图标样式
import "bootstrap/dist/css/bootstrap.css"
import "./assets/fonts/iconfont.css"
创建/复制如下文件
从昨天案例中-直接复制过来components/MyHeader.vue
components/MyTabBar.vue
views/MyGoodsList.vue
views/MyGoodsSearch.vue
views/MyUserInfo.vue
components/MyTable.vue
目标: 实现MyTabBar.vue组件
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cf3MTuW7-1625832204072)(images/image-20210511172826513.png)]
分析:
①: 基本标签+样式(md里复制)
②: 为tabbar组件指定数据源
③: 数据源最少2个, 最多5个(validator)
④: 从App.vue给MyTabBar.vue传入底部导航的数据
⑤: MyTabBar.vue中循环展示
App.vue-数组准备
tabList: [
{
iconText: "icon-shangpinliebiao",
text: "商品列表",
componentName: "MyGoodsList"
},
{
iconText: "icon-sousuo",
text: "商品搜索",
componentName: "MyGoodsSearch"
},
{
iconText: "icon-user",
text: "我的信息",
componentName: "MyUserInfo"
}
]
MyTabBar.vue - 标签模板
MyTabBar.vue正确代码(不可复制)
不要忘了把tabList数组从App.vue -> MyTabBar.vue
目标: 点击底部导航实现高亮效果
分析:
①: 绑定点击事件, 获取点击的索引
②: 循环的标签设置动态class, 遍历的索引, 和点击保存的索引比较, 相同则高亮
效果演示:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4RONsF4y-1625832204073)(images/image-20210511173026688.png)]
MyTabBar.vue(正确代码)
目的: 点击底部导航, 切换页面组件显示
需求: 点击底部切换组件
分析:
①: 底部导航传出动态组件名字符串到App.vue
②: 切换动态组件is属性的值为要显示的组件名
效果演示:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jTRSRhz4-1625832204074)(images/tabbar_切换组件.gif)]
补充: 给内容div.app- 设置上下内边距
App.vue - 引入并注册
MyTabBar.vue - 点击传递过来组件名
methods: {
btn(index, theObj) {
this.selIndex = index; // 点谁, 就把谁的索引值保存起来
this.$emit("changeCom", theObj.componentName); // 要切换的组件名传App.vue
},
},
目标: 为MyGoodsList页面, 准备表格组件MyTable.vue-铺设展示数据
分析:
①: 封装MyTable.vue – 准备标签和样式
②: axios在MyGoodsList.vue请求数据回来
③: 请求地址: https://www.escook.cn/api/goods
④: 传入MyTable.vue中循环数据显示
⑤: 给删除按钮添加bootstrap的样式: btn btn-danger btn-sm
效果演示:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-h9klruKe-1625832204076)(images/image-20210511173412963.png)]
MyTable.vue - 准备table整个表格标签和样式(可复制)
#
商品名称
价格
标签
操作
1
商品
998
xxx
xxx
使用axios请求数据, 把表格页面铺设出来
main.js - 注册axios配置默认地址
import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";
MyGoodsList.vue - 使用axios请求数据, 把数据传入给MyTable.vue里循环铺设
MyTable.vue里正确代码(不可复制)
#
商品名称
价格
标签
操作
{{ obj.id }}
{{ obj.goods_name }}
{{ obj.goods_price }}
{{ obj.tags }}
目标: 使用插槽技术, 和作用域插槽技术, 给MyTable.vue组件, 自定义列标题, 自定义表格内容
分析:
①: 把MyTable.vue里准备slot
②: 使用MyTable组件时传入具体标签
步骤:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-C4mNyRNE-1625832204078)(images/image-20201225230419844-1610538206980.png)]
MyTable.vue - 留好具名插槽
MyGoodsList.vue 使用
#
商品名称
价格
标签
操作
{{ scope.row.id }}
{{ scope.row.goods_name }}
{{ scope.row.goods_price }}
{{ scope.row.tags }}
目标: 把单元格里的标签, tags徽章铺设下
分析:
①: 插槽里传入的td单元格
②: 自定义span标签的循环展示-给予样式
效果演示:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mPAn0rW3-1625832204079)(images/image-20210511174218209.png)]
bootstrap徽章: https://v4.bootcss.com/docs/components/badge/
MyGoodsList.vue - 插槽
{{ str }}
下面额外添加样式
目标: 点击删除对应这条数据
分析:
①: 删除按钮绑定点击事件
②: 作用域插槽绑定id值出来
③: 传给删除方法, 删除MyGoodsList.vue里数组里数据
效果演示
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Mx3YbJyF-1625832204080)(images/tabbar_删除功能.gif)]
提示: id在MyTable.vue里, 但是MyGoodsList.vue里要使用, 而且在插槽位置, 使用作用域插槽已经把整个obj对象(包含id)带出来了
MyTable.vue
2. my-goods-list.vue
根据 id 删除
removeBtn(id){
let index = this.list.findIndex(obj => obj.id === id)
this.list.splice(index, 1)
},
目标: 实现点击tab按钮, 出现输入框自动获取焦点, 失去焦点关闭input, 回车新增tag, esc清空内容
效果目标:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Jcn962w5-1625832204081)(images/tabbar_tab功能.gif)]
MyGoodsList.vue - 标签位置添加
注意: 每个tab按钮和input都是独立变量控制, 那么直接在row身上的属性控制即可
main.js - 定义全局自定义指令
// 全局指令
Vue.directive("focus", {
inserted(el){
el.focus()
}
})
MyGoodsList.vue - 使用 v-focus指令
监听input失去焦点事件, 让input消失
@blur="scope.row.inputVisible = false"
监听input的回车事件, 如果无数据拦截代码
@keydown.enter="enterFn(scope.row)"
事件处理函数
enterFn(obj){ // 回车
// console.log(obj.inputValue);
if (obj.inputValue.trim().length === 0) {
alert('请输入数据')
return
}
obj.tags.push(obj.inputValue) // 表单里的字符串状态tags数组
obj.inputValue = ""
}
@keydown.esc="scope.row.inputValue = ''"
使用方式:当组件当做标签进行使用的时候,用slot可以用来接受组件标签包裹的内容,当给solt标签添加name属性的 时候,可以调换响应的位置
(高级用法) 插槽作用域: 当传递的不是单一的标签, 例如需要循环时, 把要循环的标签传入, 组件内使用v-for在slot标签上, 内部可以v-bind:把值传出来, 再外面把值赋予进去, 看示例
<current-user>
<template v-slot:default="slotProps">
{{ slotProps.user.firstName }}
</template>
</current-user>
// current-user组件, user属性和值, 绑定给slotProps上
<span>
<slot v-bind:user="user">
{{ user.lastName }}
</slot>
</span>
扩展阅读: https://cn.vuejs.org/v2/guide/components-slots.html (了解即可, 一般用不上)
1**)前言:**在开发Vue项目的时候,大部分组件是没必要多次渲染的,所以Vue提供了一个内置组件keep-alive来缓存组件内部状态,避免重新渲染,在开发Vue项目的时候,大部分组件是没必要多次渲染的,所以Vue提供了一个内置组件keep-alive来缓存组件内部状态,避免重新渲染
2**)生命周期函数:**在被keep-alive包含的组件/路由中,会多出两个生命周期的钩子:activated 与 deactivated。
1**、activated钩子:**在在组件第一次渲染时会被调用,之后在每次缓存组件被激活时调用。
2**、Activated钩子调用时机:** 第一次进入缓存路由/组件,在mounted后面,beforeRouteEnter守卫传给 next 的回调函数之前调用,并且给因为组件被缓存了,再次进入缓存路由、组件时,不会触发这些钩子函数,beforeCreate created beforeMount mounted 都不会触发
1**、deactivated钩子:**组件被停用(离开路由)时调用。
2**、deactivated钩子调用时机**:使用keep-alive就不会调用beforeDestroy(组件销毁前钩子)和destroyed(组件销毁),因为组件没被销毁,被缓存起来了,这个钩子可以看作beforeDestroy的替代,如果你缓存了组件,要在组件销毁的的时候做一些事情,可以放在这个钩子里,组件内的离开当前路由钩子beforeRouteLeave => 路由前置守卫 beforeEach =>全局后置钩子afterEach => deactivated 离开缓存组件 => activated 进入缓存组件(如果你进入的也是缓存路由)
全局定义指令:在vue对象的directive方法里面有两个参数,一个是指令名称,另外一个是函数。组件内定义指令:directives
钩子函数:bind(绑定事件触发)、inserted(节点插入的时候触发)、update(组件内相关更新)
钩子函数参数:el、binding
1**)动态组件**
, componentName可以是在本页面已经注册的局部组件名和全局组件名,也可以是一个组件的选项对象。 当控制componentName改变时就可以动态切换选择组件。
2**)is的用法**
有些HTML元素,诸如
而有些HTML元素,诸如
所以上面会被作为无效的内容提升到外部,并导致最终渲染结果出错。应该这么写:
目的: 封装一个复用的组件, 可以动态的插入标签, 来作为注册页的一块项
代码:
<div id="app">
<child-com :title="'姓名'">
<input type='text'/>
child-com>
<child-com :title="'密码'">
<input type='password' />
child-com>
<child-com :title="'性别'">
<input type='radio' name="sex" value="男"/>男
<input type='radio' name="sex" value="女"/>女
child-com>
<child-com :title="'爱好'">
<input type='checkbox' value="抽烟" />抽烟
<input type='checkbox' value="喝酒" />喝酒
<input type='checkbox' value="烫头" />烫头
child-com>
<child-com :title="'来自于'">
<select>
<option value="北京">北京option>
<option value="天津">天津option>
<option value="南京">南京option>
select>
child-com>
div>
<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js">script>
<script>
new Vue({
el: "#app",
components: {
childCom: { // 组件名字
props: {
title: {
type: String
}
},
template: `
{{title}}
`
}
}
})
script>