Vue组件-进阶
一、学习目标
- 能够掌握
vue组件进阶知识使用
- 能够掌握
自定义指令
- 能够完成
tabbar案例
二、Vue组件进阶
1. 动态组件
目标:多个组件使用同一个挂载点,并动态切换,这就是动态组件
需求:完成一个注册功能页面,2个按钮切换,一个填写注册信息,一个填写用户简介信息
- 准备被切换的 - UserName.vue / UserInfo.vue2个组件
- 引入到UseDynamic.vue注册
- 准备变量来承载要显示的"组件名"
- 设置挂载点使用is属性来设置要显示那个组件
- 点击按钮-修改comName变量里的"组件名"
<template>
<div>
<button @click="comName = 'UserName'">账号密码填写</button>
<button @click="comName = 'UserInfo'">个人信息填写</button>
<p>下面显示注册组件-动态切换:</p>
<div style="border: 1px solid red;">
<component :is="comName"></component>
</div>
</div>
</template>
<script>
// 目标: 动态组件 - 切换组件显示
// 场景: 同一个挂载点要切换 不同组件 显示
// 1. 创建要被切换的组件 - 标签+样式
// 2. 引入到要展示的vue文件内, 注册
// 3. 变量-承载要显示的组件名
// 4. 设置挂载点<component :is="变量"></component>
// 5. 点击按钮-切换comName的值为要显示的组件名
import UserName from '../components/01/UserName'
import UserInfo from '../components/01/UserInfo'
export default {
data(){
return {
comName: "UserName"
}
},
components: {
UserName,
UserInfo
}
}
</script>
在App.vue引入UseDynaminc.vue并使用显示
总结:vue内置component组件,配合is属性,设置要显示的组件名字
2. 组件缓存
目标:组件切换会导致组件被频繁销毁和重新创建,性能不高
使用Vue内置的keep-alive组件,可以让包裹的组件报错在内存中不被销毁
- 可以先给UserName.vue和UserInfo.vue注册created和destroyed声明周期事件,观察创建和销毁过程
- 使用keep-alive内置的vue组件,让动态组件缓存而不是销毁
语法:
Vue内置的keep-alive组件包起来要频繁切换的组件
<div style="border: 1px solid red;">
<!-- Vue内置keep-alive组件, 把包起来的组件缓存起来 -->
<keep-alive>
<component :is="comName"></component>
</keep-alive>
</div>
补充生命周期:
- activated - 激活
- deactivated - 失去激活状态
总结:keep-alive可以提高组件的性能,内部包裹的标签不会被销毁和重新创建,触发激活和非激活的生命周期方法
3. 激活和非激活
目标:被缓存的组件不能创建和销毁,而是激活和非激活
- activated - 激活时触发
- deactivated - 失去激活状态触发
4. 组件插槽
目标:用于实现组件的内容分发,通过slot标签,可以接收到写在组件标签内的内容
vue提供组件插槽能力,允许开发者在封装组件时,把不确定的部分定义为插槽
需求:以前折叠面板案例,想要实现不同内容显示,我们把折叠面板里的Pannel组件,添加组件插槽方式
语法口诀:
- 组件内用
- 使用组件时夹着的地方,转入标签替换slot
Pannel.vue - 组件(直接复制)
<template>
<div>
<!-- 按钮标题 -->
<div class="title">
<h4>芙蓉楼送辛渐</h4>
<span class="btn" @click="isShow = !isShow">
{{ isShow ? "收起" : "展开" }}
</span>
</div>
<!-- 下拉内容 -->
<div class="container" v-show="isShow">
<p>寒雨连江夜入吴,</p>
<p>平明送客楚山孤。</p>
<p>洛阳亲友如相问,</p>
<p>一片冰心在玉壶。</p>
</div>
</div>
</template>
<script>
export default {
data() {
return {
isShow: false,
};
},
};
</script>
<style scoped>
h3 {
text-align: center;
}
.title {
display: flex;
justify-content: space-between;
align-items: center;
border: 1px solid #ccc;
padding: 0 1em;
}
.title h4 {
line-height: 2;
margin: 0;
}
.container {
border: 1px solid #ccc;
padding: 0 1em;
}
.btn {
/* 鼠标改成手的形状 */
cursor: pointer;
}
img {
width: 50%;
}
</style>
UserSlot.vue - 使用组件
<template>
<div id="container">
<div id="app">
<h3>案例:折叠面板</h3>
</div>
</div>
</template>
<script>
export default {
};
</script>
<style>
#app {
width: 400px;
margin: 20px auto;
background-color: #fff;
border: 4px solid blueviolet;
border-radius: 1em;
box-shadow: 3px 3px 3px rgba(0, 0, 0, 0.5);
padding: 1em 2em 2em;
}
</style>
UseSlot.vue - 组件插槽使用
<template>
<div id="container">
<div id="app">
<h3>案例:折叠面板</h3>
<Pannel>
<img src="../assets/mm.gif" alt="">
<span>我是内容</span>
</Pannel>
<Pannel>
<p>寒雨连江夜入吴,</p>
<p>平明送客楚山孤。</p>
<p>洛阳亲友如相问,</p>
<p>一片冰心在玉壶。</p>
</Pannel>
<Pannel></Pannel>
</div>
</div>
</template>
<script>
import Pannel from "../components/03/Pannel";
export default {
components: {
Pannel,
},
};
</script>
总结:组件内容分发技术,slot占位,使用组件时传入替换slot位置的标签
5. 插槽默认内容
目标:如果外面不给传,想给个默认显示内容
口诀:夹着内容默认显示内容,如果不给插槽slot传东西,则使用夹着的内容在原地显示
<slot>默认内容</slot>
6. 具名插槽
目标:当一个组件内容有2处以上需要外部传入标签的地方
传入的标签可以分别派发给不同的slot位置
要求:v-slot一般用跟template标签使用(template是html5新出标签内容模板元素,不会渲染到页面上,一般被vue解析内部标签)
04/Pannel.vue - 留下具名slot
<template>
<div>
<!-- 按钮标题 -->
<div class="title">
<slot name="title"></slot>
<span class="btn" @click="isShow = !isShow">
{{ isShow ? "收起" : "展开" }}
</span>
</div>
<!-- 下拉内容 -->
<div class="container" v-show="isShow">
<slot name="content"></slot>
</div>
</div>
</template>
04_UseSlot.vue使用
<template>
<div id="container">
<div id="app">
<h3>案例:折叠面板</h3>
<Pannel>
<template v-slot:title>
<h4>芙蓉楼送辛渐</h4>
</template>
<template v-slot:content>
<img src="../assets/mm.gif" alt="">
<span>我是内容</span>
</template>
</Pannel>
<Pannel>
<template #title>
<span style="color: red;">我是标题</span>
</template>
<template #content>
<p>寒雨连江夜入吴,</p>
<p>平明送客楚山孤。</p>
<p>洛阳亲友如相问,</p>
<p>一片冰心在玉壶。</p>
</template>
</Pannel>
</div>
</div>
</template>
<script>
import Pannel from "../components/04/Pannel";
export default {
components: {
Pannel,
},
};
</script>
v-slot可以简化成#使用
b-bind可以省略成:v-on:可以省略成@ 那么v-slot:可以简化成#
总结:slot的name属性起插槽名,使用组件时,template配合#插槽名传入具体标签
7. 作用域插槽
目标:子组件里值,在给插槽赋值时在父组件环境下使用
复习:插槽内slot中显示默认内容
例子:默认内容在子组件中,但是父亲在给插槽传值,想要改变插槽显示默认内容
口诀:
- 子组件,在slot上绑定属性和子组件内的值
- 使用组件,传入自定义标签,用template和v-slot=“自定义变量名”
- scope变量名自动绑定slot上所有属性和值
05/Pannel.vue - 定义组件, 和具名插槽, 给slot绑定属性和值
<template>
<div>
<!-- 按钮标题 -->
<div class="title">
<h4>芙蓉楼送辛渐</h4>
<span class="btn" @click="isShow = !isShow">
{{ isShow ? "收起" : "展开" }}
</span>
</div>
<!-- 下拉内容 -->
<div class="container" v-show="isShow">
<slot :row="defaultObj">{{ defaultObj.defaultOne }}</slot>
</div>
</div>
</template>
<script>
// 目标: 作用域插槽
// 场景: 使用插槽, 使用组件内的变量
// 1. slot标签, 自定义属性和内变量关联
// 2. 使用组件, template配合v-slot="变量名"
// 变量名会收集slot身上属性和值形成对象
export default {
data() {
return {
isShow: false,
defaultObj: {
defaultOne: "无名氏",
defaultTwo: "小传同学"
}
};
},
};
</script>
05_UseSlot.vue
<template>
<div id="container">
<div id="app">
<h3>案例:折叠面板</h3>
<Pannel>
<!-- 需求: 插槽时, 使用组件内变量 -->
<!-- scope变量: {row: defaultObj} -->
<template v-slot="scope">
<p>{{ scope.row.defaultTwo }}</p>
</template>
</Pannel>
</div>
</div>
</template>
<script>
import Pannel from "../components/05/Pannel";
export default {
components: {
Pannel,
},
};
</script>
总结:组件内变量绑定在slot上,然后使用组件v-slot="变量"变量上就会绑定slot身上属性和值
8. 作用域插槽使用场景
目标:了解作用域插槽使用场景,自定义组件内标签 + 内容
案例:封装一个表格组件,在表格组件内循环产生单元格
准备MyTable.vue组件 - 内置表格,传入数组循环铺设页面,把对象每个内容显示在单元格里
准备UserTable.vue - 准备数据传入给MyTable.vue使用
06/MyTable.vue - 模板
<template>
<div>
<table border="1">
<thead>
<tr>
<th>序号</th>
<th>姓名</th>
<th>年龄</th>
<th>头像</th>
</tr>
</thead>
<thead>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</thead>
</table>
</div>
</template>
<script>
export default {
}
</script>
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标签
正确做法:
在MyTable.vue的td中准备占位,但是外面需要把图片地址赋予给src属性,所以在slot上把obj数据绑定
06/MyTable.vue
<template>
<div>
<table border="1">
<thead>
<tr>
<th>序号</th>
<th>姓名</th>
<th>年龄</th>
<th>头像</th>
</tr>
</thead>
<tbody>
<tr v-for="(obj, index) in arr" :key="index">
<td>{{ index + 1 }}</td>
<td>{{ obj.name }}</td>
<td>{{ obj.age }}</td>
<td>
<slot :row="obj">
<!-- 默认值给上,如果使用组件不自定义标签显示默认文字 -->
{{ obj.headImgUrl}}
</slot>
</td>
</tr>
</tbody>
</table>
</div>
</template>
<script>
export default {
props: {
arr: Array
}
}
</script>
在UseTable使用MyTable的时候, template上v-slot绑定变量, 传入img组件设置图片地址
<template>
<div>
<MyTable :arr="list"></MyTable>
<MyTable :arr="list">
<!-- scope: {row: obj} -->
<template v-slot="scope">
<a :href="scope.row.headImgUrl">{{ scope.row.headImgUrl }}</a>
</template>
</MyTable>
<MyTable :arr="list">
<template v-slot="scope">
<img style="width: 100px;" :src="scope.row.headImgUrl" alt="">
</template>
</MyTable>
</div>
</template>
<script>
import MyTable from "../components/06/MyTable";
export default {
components: {
MyTable,
},
data() {
return {
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",
},
],
};
},
};
</script>
<style>
</style>
三、自定义指令
自定义指令文档:https://www.vue3js.cn/docs/zh/guide/custom-directive.html
除了核心功能默认内置的指令(v-model
和 v-show
),
Vue也允许注册自定义指令。v-xxx
html + css 的复用的主要形式是组件
你需要对普通DOM元素进行底层封装,这时候就会用到自定义指令
1. 自定义指令 - 注册
目标:获取标签,扩展额外的功能
1.1 局部注册和使用
UseDirective.vue - 只能在当前组件.vue文件中使用
<template>
<div>
<!-- <input type="text" v-gfocus> -->
<input type="text" v-focus>
</div>
</template>
<script>
// 目标: 创建 "自定义指令", 让输入框自动聚焦
// 1. 创建自定义指令
// 全局 / 局部
// 2. 在标签上使用自定义指令 v-指令名
// 注意:
// inserted方法 - 指令所在标签, 被插入到网页上触发(一次)
// update方法 - 指令对应数据/标签更新时, 此方法执行
export default {
data(){
return {
colorStr: 'red'
}
},
directives: {
focus: {
inserted(el){
el.focus()
}
}
}
}
</script>
<style>
</style>
1.2 全局注册
在main.js用 Vue.directive()方法来进行注册, 以后随便哪个.vue文件里都可以直接用v-fofo指令
// 全局指令 - 到处"直接"使用
Vue.directive("gfocus", {
inserted(el) {
el.focus() // 触发标签的事件方法
}
})
总结:全局组成自定义指令,哪里都能用,局部注册,只能在当前vue文件里用
2. 传值
目标:使用自定义指令,传入一个值
需求:定义color指令 - 传入一个颜色,给标签设置文字颜色
main.js定义出修改一下
// 目标: 自定义指令传值
Vue.directive('color', {
inserted(el, binding) {
el.style.color = binding.value
},
update(el, binding) {
el.style.color = binding.value
}
})
Direct.vue处更改一下
<p v-color="colorStr" @click="changeColor">修改文字颜色</p>
<script>
data() {
return {
theColor: "red",
};
},
methods: {
changeColor() {
this.theColor = 'blue';
},
},
</script>
总结:v-xxx,自定义指令,获取元素DOM,自定义操作
四、案例-tabbar
完成如下案例和各步功能
知识点:
- 组件封装
- 动态组件
- keep-alive
- 作用域插槽
- 自定义指令
1. 项目初始化
目标:创建项目文件夹,引入字体图标,下载bootstrap,less,less-loader@5.0.0,axios,在App.vue注册组件
- 需求:从0新建项目,拆分组件,创建使用
组件分析:
- 组件拆分:
- MyHeader.vue - 复用之前的
- MyTabBar.vue - 底部导航
- MyTable.vue - 封装表格
- 三个页面
- MyGoodsList.vue - 商品页
- MyGoodsSearch.vue - 搜索页
- MyUserInfo.vue - 用户信息页
思路分析:
- vue create tabbar-demo
- yarn add less less-loader@5.0.0 -D
- yarn add bootstrap axios 并在main.js引入和全局属性
- 根据需求 - 创建需要的页面组件
- 把购物车案例 - 封装的MyHeader.vue文件复制过来复用
- 从App.vue - 引入组织相关标签
新建工程:
vue create tabbar-demo
yarn add less less-loader@5.0.0 -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
2. 底部封装
目标:实现MyTabBar.vue组件
- 需求:把底部导航也灵活封装起来
分析:
- 基本标签 + 样式
- 为tabbar组件指定数据源
- 数据源最少2个,最多5个
- 从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 - 标签模板
<template>
<div class="my-tab-bar">
<div class="tab-item">
<!-- 图标 -->
<span class="iconfont"></span>
<!-- 文字 -->
<span></span>
</div>
</div>
</template>
<script>
export default {
}
</script>
<style lang="less" scoped>
.my-tab-bar {
position: fixed;
left: 0;
bottom: 0;
width: 100%;
height: 50px;
border-top: 1px solid #ccc;
display: flex;
justify-content: space-around;
align-items: center;
background-color: white;
.tab-item {
display: flex;
flex-direction: column;
align-items: center;
}
}
.current {
color: #1d7bff;
}
</style>
MyTabBar.vue
<template>
<div class="my-tab-bar">
<div
class="tab-item"
v-for="(obj, index) in arr"
:key="index"
>
<!-- 图标 -->
<span class="iconfont" :class="obj.iconText"></span>
<!-- 文字 -->
<span>{{ obj.text }}</span>
</div>
</div>
</template>
<script>
export default {
props: {
arr: {
type: Array,
required: true,
// 自定义校验规则
validator(value) {
// value就是接到数组
if (value.length >= 2 && value.length <= 5) {
return true; // 符合条件就return true
} else {
console.error("数据源必须在2-5项");
return false;
}
},
},
}
};
</script>
不要忘了把tabList数组从App.vue -> MyTabBar.vue
3. 底部高亮
目标:点击底部导航实现高亮效果
- 需求:点击底部实现高亮效果
分析:
- 绑定点击事件,获取点击的索引
- 循环的标签设置动态class,遍历的索引,和点击保存的索引比较,相同则高亮
效果演示:
MyTabBar.vue
<template>
<div class="my-tab-bar">
<div class="tab-item"
v-for="(obj, index) in arr"
:key="index"
:class="{current: activeIndex === index}"
@click="activeIndex = index">
<!-- 图标 -->
<span class="iconfont" :class="obj.iconText"></span>
<!-- 文字 -->
<span>{{ obj.text }}</span>
</div>
</div>
</template>
<script>
export default {
data(){
return {
activeIndex: 0 // 高亮元素下标
}
},
// ....其他代码
};
</script>
4. 组件切换
目的:点击底部导航,切换页面组件显示
- 需求:点击底部切换组件
- 分析:
- 底部导航传出动态组件名字字符串到App.vue
- 切换动态组件is属性的值为要显示的组件名
- 效果图:
补充:给内容div.app - 设置上下内边距
App.vue - 引入并注册
<template>
<div>
<MyHeader
:background="'blue'"
:fontColor="'white'"
title="TabBar案例"
></MyHeader>
<div class="main">
<component :is="comName"></component>
</div>
<MyTabBar :arr="tabList"
@changeCom="changeComFn"
></MyTabBar>
</div>
</template>
<script>
import MyHeader from "./components/MyHeader";
// 目标: 完成底部封装
// 1. MyTabBar.vue 组件标签+样式 准备
// 2. 字体图标引入
// 3. 准备底部数据
// 4. 使用MyTabBar组件, 传入数据(父->子), 循环产生底部导航
// 5. 子组件内props自定义检验规则(2-5项)
// 6. 子组件内循环产生底部导航
import MyTabBar from './components/MyTabBar'
// 目标: 切换组件显示
// 1. 创建组件 - 编写内容
// 2. 引入App.vue注册
// 3. 挂载点设置is
// 4. 切换comName的值(重要)
// 5. 底部导航点击- MyTabBar.vue里
// 子 -> 父技术 (传要切换的组件名出来)
import MyGoodsList from './views/MyGoodsList'
import MyGoodsSearch from './views/MyGoodsSearch'
import MyUserInfo from './views/MyUserInfo'
export default {
data() {
return {
comName: "MyGoodsList", // 默认显示的组件
tabList: [ // 底部导航的数据
{
iconText: "icon-shangpinliebiao",
text: "商品列表",
componentName: "MyGoodsList",
},
{
iconText: "icon-sousuo",
text: "商品搜索",
componentName: "MyGoodsSearch",
},
{
iconText: "icon-user",
text: "我的信息",
componentName: "MyUserInfo",
},
],
};
},
components: {
MyHeader,
MyTabBar,
MyGoodsList,
MyGoodsSearch,
MyUserInfo
},
methods: {
changeComFn(cName){
this.comName = cName; // MyTabBar里选出来的组件名赋予给is属性的comName
// 导致组件的切换
}
}
};
</script>
<style scoped>
.main{
padding-top: 45px;
padding-bottom: 51px;
}
</style>
MyTabBar.vue - 点击传递过来组件名
methods: {
btn(index, theObj) {
this.selIndex = index; // 点谁, 就把谁的索引值保存起来
this.$emit("changeCom", theObj.componentName); // 要切换的组件名传App.vue
},
},
5. 商品列表
目标:为MyGoodsList页面,准备表格组件MyTable.vue - 铺设展示数据
-
需求:商品列表铺设页面
-
分析:
- 封装MyTable.vue - 准备标签和样式
- axios在MyGoodsList.vue请求数据回来
- 请求地址:https://www.escook.cn/api/goods
- 传入MyTable.vue中循环数据显示
- 给删除按钮添加bootstrap的样式:btn btn-danger btn-sm
-
效果演示:
MyTable.vue - 准备table整个表格标签和样式
<template>
<table class="table table-bordered table-stripped">
<!-- 表格标题区域 -->
<thead>
<tr>
<th>#</th>
<th>商品名称</th>
<th>价格</th>
<th>标签</th>
<th>操作</th>
</tr>
</thead>
<!-- 表格主体区域 -->
<tbody>
<tr >
<td>1</td>
<td>商品</td>
<td>998</td>
<td>xxx</td>
<td>xxx</td>
</tr>
</tbody>
</table>
</template>
<script>
export default {
name: 'MyTable'
}
</script>
<style scoped lang="less">
.my-goods-list {
.badge {
margin-right: 5px;
}
}
</style>
使用axios请求数据,把表格页面铺设出来
main.js - 注册axios配置默认地址
import axios from 'axios'
axios.default.baseURL = 'https://www.escook.cn'
Vue.prototype.$axios = axios
MyGoodsList.vue - 使用axios请求数据, 把数据传入给MyTable.vue里循环铺设
<template>
<div>
<MyTable :arr="list">
</MyTable>
</div>
</template>
<script>
import MyTable from "../components/MyTable";
import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";
// 目标: 循环商品列表表格
// 1. 封装MyTable.vue 整体表格组件-一套标签和样式
// 2. axios请求数据
// 3. 传入MyTable组件里循环tr显示数据
// 目标: 展示tags标签
// 1. tags数组 - 某个td循环span使用文字
// 2. span设置bs的样式
// 目标: 删除数据
// 1. 删除按钮 - 点击事件
// 2. 作用域插槽把索引值关联出来了
// scope身上就有row和index
// 3. 删除中使用scope.index的索引值
// 4. 删除事件里删除数组里对应索引值的数据
export default {
components: {
MyTable,
},
data() {
return {
list: [] // 所有数据
};
},
created() {
axios({
url: "/api/goods",
}).then((res) => {
console.log(res);
this.list = res.data.data;
});
}
};
</script>
MyTable.vue
<template>
<table class="table table-bordered table-stripped">
<!-- 表格标题区域 -->
<thead>
<tr>
<th>#</th>
<th>商品名称</th>
<th>价格</th>
<th>标签</th>
<th>操作</th>
</tr>
</thead>
<!-- 表格主体区域 -->
<tbody>
<tr v-for="(obj, index) in arr"
:key="obj.id"
>
<td>{{ obj.id }}</td>
<td>{{ obj.goods_name }}</td>
<td>{{ obj.goods_price }}</td>
<td>{{ obj.tags }}</td>
<td>
<button class="btn btn-danger btn-sm">删除</button>
</td>
</tr>
</tbody>
</table>
</template>
<script>
export default {
name: 'MyTable',
props: {
arr: Array
}
}
</script>
<style scoped lang="less">
.my-goods-list {
.badge {
margin-right: 5px;
}
}
</style>
6. 商品列表 - 插槽
目标:使用插槽技术,和作用域插槽技术,给MyTable.vue组件,自定义列标题,自定义表格内容
- 需求:允许用户自定义表格头和表格表单元格内容
- 分析:
- 把MyTable.vue里准备slot
- 使用MyTable组件时传入具体标签
- 步骤:
- 提高组件 - 复用性和灵活性,把表格列标题thead部分预留标签,设置name属性
- 使用MyTable.vue时,传入列标题标签
- 表格内容td部分也可以让组件使用自定义,也给tbody下tr内留好标签和name属性名
- 使用插槽需要用到插槽内的obj对象上的数据,使用作用域插槽技术
MyTable.vue - 留好具名插槽
<template>
<table class="table table-bordered table-stripped">
<!-- 表格标题区域 -->
<thead>
<tr>
<!-- <th>#</th>
<th>商品名称</th>
<th>价格</th>
<th>标签</th>
<th>操作</th> -->
<slot name="header"></slot>
</tr>
</thead>
<!-- 表格主体区域 -->
<tbody>
<tr v-for="(obj, index) in arr"
:key="obj.id"
>
<!-- <td>{{ obj.id }}</td>
<td>{{ obj.goods_name }}</td>
<td>{{ obj.goods_price }}</td>
<td>{{ obj.tags }}</td>
<td>
<button class="btn btn-danger btn-sm">删除</button>
</td> -->
<slot name="body" :row="obj" :index="index"></slot>
</tr>
</tbody>
</table>
</template>
<script>
export default {
name: 'MyTable',
props: {
arr: Array
}
}
</script>
MyGoodsList.vue 使用
<template>
<div>
<MyTable :arr="list">
<template #header>
<th>#</th>
<th>商品名称</th>
<th>价格</th>
<th>标签</th>
<th>操作</th>
</template>
<!-- scope的值: {row: obj, index: 索引值} -->
<template #body="scope">
<td>{{ scope.row.id }}</td>
<td>{{ scope.row.goods_name }}</td>
<td>{{ scope.row.goods_price }}</td>
<td>
{{ scope.row.tags }}
</td>
<td>
<button class="btn btn-danger btn-sm"
>删除</button>
</td>
</template>
</MyTable>
</div>
</template>
<script>
import MyTable from "../components/MyTable";
import axios from "axios";
axios.defaults.baseURL = "https://www.escook.cn";
// 目标: 循环商品列表表格
// 1. 封装MyTable.vue 整体表格组件-一套标签和样式
// 2. axios请求数据
// 3. 传入MyTable组件里循环tr显示数据
// 目标: 展示tags标签
// 1. tags数组 - 某个td循环span使用文字
// 2. span设置bs的样式
// 目标: 删除数据
// 1. 删除按钮 - 点击事件
// 2. 作用域插槽把索引值关联出来了
// scope身上就有row和index
// 3. 删除中使用scope.index的索引值
// 4. 删除事件里删除数组里对应索引值的数据
export default {
components: {
MyTable,
},
data() {
return {
list: [] // 所有数据
};
},
created() {
axios({
url: "/api/goods",
}).then((res) => {
console.log(res);
this.list = res.data.data;
});
}
};
</script>
<style>
</style>
7. 商品列表 - tags微标
目标:把单元格里的标签,tags徽章铺设下
-
需求:标签列自定义显示
-
分析:
- 插槽里传入的td单元格
- 自定义span标签的循环展示 - 给予样式
-
效果演示;
bootstrap徽章: https://v4.bootcss.com/docs/components/badge/
MyGoodsList.vue - 插槽
<span v-for="(str, ind) in scope.row.tags" :key="ind"
class="badge badge-warning"
>
{{ str }}
</span>
下面额外添加样式
<style lang="less" scoped>
.my-goods-list {
.badge {
margin-right: 5px;
}
}
</style>
8. 商品列表 - 删除功能
目标:点击删除对应这条数据
-
需求:点击删除按钮删除数据
-
分析:
- 删除按钮绑定点击事件
- 作用域插槽绑定id值出来
- 传给删除方法,删除MyGoodsList.vue里数组里数据
-
效果演示:
提示:id在MyTable.vue里,但是 MyGoodsList.vue里要使用,而且在插槽位置,使用作用域插槽已经把整个obj对象(包含id)带出来了
MyTable.vue
<slot name="body" :row="obj"></slot>
-
MyGoodsList.vue - 注册点击事件
<button class="btn btn-danger btn-sm" @click="removeBtn(scope.row.id)" >删除</button>
-
my-goods-list.vue
根据 id 删除removeBtn(id){ let index = this.list.findIndex(obj => obj.id === id) this.list.splice(index, 1) },
9. 添加tab
目标:实现点击tab按钮,出现输入框自动获取焦点,失去焦点关闭input,回车新增tag,esc清空内容
- 需求1:点击Tab,按钮消失,输入出现
- 需求2:输入框自动聚焦
- 需求3:失去焦点,输入框消失,按钮出
- 需求4:检测input回车,无数据拦截
- 需求5:检测input取消,清空数据
- 需求6:检测input回车,有数据添加
效果目标:
9.1 点击按钮消失,输入框出现
MyGoodsList.vue - 标签位置添加
注意: 每个tab按钮和input都是独立变量控制, 那么直接在row身上的属性控制即可
<input
class="tag-input form-control"
style="width: 100px;"
type="text"
v-if="scope.row.inputVisible"
/>
<button
v-else
style="display: block;"
class="btn btn-primary btn-sm add-tag"
@click="scope.row.inputVisible = true"
>+Tag</button>
9.2 input自动获取焦点
main.js - 定义全局自定义指令
// 全局指令
Vue.directive('focus',{
inserted(el) {
el.focus()
}
})
MyGoodsList.vue - 使用 v-focus指令
9.3 input失去焦点关闭input
监听input失去焦点事件,让input消失
@blur="scope.row.inputVisible = false"
9.4 input回车新增tag
监听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 = ""
}
9.5 input框esc清空内容
@keydown.esc="scope.row.inputValue = ''"
总结
- 动态组件的使用步骤
- 组件缓存使用步骤和作用
- 组件插槽默认使用
- 插槽默认显示的内容
- 多个插槽时,具名插槽如何使用
- 作用插槽如何使用以及目的
- 自定义指令如何使用
,失去焦点关闭input,回车新增tag,esc清空内容
- 需求1:点击Tab,按钮消失,输入出现
- 需求2:输入框自动聚焦
- 需求3:失去焦点,输入框消失,按钮出
- 需求4:检测input回车,无数据拦截
- 需求5:检测input取消,清空数据
- 需求6:检测input回车,有数据添加
效果目标:
[外链图片转存中…(img-2WMjORkV-1661392744775)]
9.1 点击按钮消失,输入框出现
MyGoodsList.vue - 标签位置添加
注意: 每个tab按钮和input都是独立变量控制, 那么直接在row身上的属性控制即可
<input
class="tag-input form-control"
style="width: 100px;"
type="text"
v-if="scope.row.inputVisible"
/>
<button
v-else
style="display: block;"
class="btn btn-primary btn-sm add-tag"
@click="scope.row.inputVisible = true"
>+Tag</button>
9.2 input自动获取焦点
main.js - 定义全局自定义指令
// 全局指令
Vue.directive('focus',{
inserted(el) {
el.focus()
}
})
MyGoodsList.vue - 使用 v-focus指令
9.3 input失去焦点关闭input
监听input失去焦点事件,让input消失
@blur="scope.row.inputVisible = false"
9.4 input回车新增tag
监听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 = ""
}
9.5 input框esc清空内容
@keydown.esc="scope.row.inputValue = ''"
总结
- 动态组件的使用步骤
- 组件缓存使用步骤和作用
- 组件插槽默认使用
- 插槽默认显示的内容
- 多个插槽时,具名插槽如何使用
- 作用插槽如何使用以及目的
- 自定义指令如何使用
- 完成tabbar案例