企业如何通过vue小程序开发满足高效运营与合规性需求
809
2022-11-26
第十章:动态组件,插槽,自定义指令
第十章:动态组件,插槽,自定义指令
本章任务
动态组件插槽自定义指令vant UI 组件库axios的配置
本章内容
动态组件自定义指令
一、动态组件
1.1 什么是动态组件
概念:动态组件就是可以动态改变的组件,之前我们引入组件,都是把组件固定写在某一个位置显示,没有办法切换组件显示,动态组件可以帮我们完成这个目的。vue中提供了一个组件
1.2 动态组件的使用
父组件中导入并注册子组件
//导入需要动态切换的 两个子组件 import com01 from '../components/Com01.vue' import com02 from '../components/Com02.vue' export default{ name:"about", //注册子组件 components:{ com01,com02 }, }
在父组件中通过
直接写is相当于直接写死 一个组件的名称,可以使用v-bind 的:is属性来完成动态的组件切换效果。 This is an about page
存在问题: 此时虽然可以动态切换组件,但是存在问题,组件的切换其实就是组件的销毁并重新创建,此时会导致再次切换回来无法保证原来的组件中的数据信息。可以通过生命周期函数验证。
com01.vue组件:
com02.vue组件:
about.vue父组件:
This is an about page
1.3 使用 keep-alive 解决上述问题
keep-alive可以保证在组件切换时 不被销毁,处于缓存状态,从而保证数据的完整性。
代码实现:使用
1.4 keep-alive的生命周期函数和属性props
1.4.1 生命周期函数
keep-alive缓存组件有两个生命周期钩子函数,分别是:
activated 组件被激活时调用deactivated 组件被缓存时调用
1.4.2 keep-alive属性
include - 字符串或正则表达式。只有名称匹配的组件会被缓存。exclude - 字符串或正则表达式。任何名称匹配的组件都不会被缓存。max - 数字。最多可以缓存多少组件实例。最多可以缓存多少组件实例。一旦这个数字达到了,在新实例被创建之前,已缓存组件中最久没有被访问的实例会被销毁掉。
**注意:**include和exclude两个属性不能同时使用。
二、插槽 slot
2.1 slot简介 v-slot
slot是插槽(槽口)的意思,其目的在于让组件的 可扩展性 更强,用来混合父组件的内容与子组件自己的模板
#例如:需要在组件中插入一些内容 //调用组件时,插入的内容会显示在模版中的这个位置 Vue.component('alert-box', { template:'#ab'})
2.2 使用场景和分类
在组件化开发中,虽然组件是一样的,但是在不同的使用场景组件的一部分内容需要有不同的内容显示
2.2.1 分类
匿名slot具名slot
2.2.2 匿名插槽
匿名插槽:从字面意思来理解是就是没有名字的插槽,特点就是可以放任何你放的东西。
**注意:**此时的插槽不是真的没有名字,而是名字默认为 ‘default’ ====>
头部区域 底部区域
2.2.3 具名插槽
假设我们的电脑主板上的各种插槽,有插CPU的,有插显卡的,有插内存的,有插硬盘的,所以假设有个组件是computer,我们不可能把显卡插到内存的位置上,具名slot也就是每个slot都有名字,不能随意替换,对应插入
vue2.6之前的版本:
你好哈哈哈哈
A paragraph for the main content.
And another one.
v-slot的简写形式就是#Here's some contact info
2.2.4 作用域插槽:借助插槽的子组件向父组件传值:
**场景:**子组件的数据需要到父组件,父组件使用子组件的数据进行一些列运算 再发送给子组件。
如果使用父子组件的通信:需要借助 自定义事件监听,props[]做数据传递,麻烦。
此时可以使用插槽来实现 子向父 再到 父向子做数据传递。
#com01.vue子组件中 #About.vue父组件中 这里是组件01
**使用场景:**作用域插槽一般在封装组件时比较常用,在封装好的组件中,通过插槽预留数据位置,在使用组件时,只需要传入需要的数据就可以了。
课下改版购物车:把加减按钮单独提取成一个组件。
vant:组件库什么是自定义指令
概念: 在vue中内部提供的有很多v-系列的指令,例如 v-if v-model 等,都有特定的作用,但是如果我们有一些特殊的需求时,vue也支持我们可以自定义指令来完成一些功能的设置。
我们自己封装一个函数,给这个函数链接一个对应的名字 把这个名字当作指令来使用。
**分类:**主要有两种
全局自定义指令局部(私有)自定义指令
3.2 为什么要用自定义指令
问题一:vue的内置指令干了什么?
就是DOM操作,vue做DOM操作的方式 跟我们之前学习 原生js 自己写的代码是一样的。
问题二:自定义指令是干啥的?
让我们自己操作DOM,按照我们自己的需求 就跟我们当年写原生js一样。 借助函数
vue的指令底层实际上就是调用函数,通过v-系列的指令调用对应的函数。
3.3 自定义指令的使用
完成案例:当输入框加载完成时,让输入框得到焦点。
全局自定义指令:
// 注册一个全局自定义指令 `v-focus`Vue.directive('focus', { // 当被绑定的元素插入到 DOM 中时…… inserted: function (el) { // 聚焦元素 el.focus() }})
局部自定义指令:
//所有的局部自定义指令放在这里。 只能再当前组件内部使用这个指令directives: { focus: { // 指令的定义 inserted: function (el) { el.focus() } }}
指令的钩子函数以及参数对象:
一个指令定义对象可以提供如下几个钩子函数 (均为可选):
bind:只调用一次,指令第一次绑定到元素时调用。在这里可以进行一次性的初始化设置。inserted:被绑定元素插入父节点时调用 (仅保证父节点存在,但不一定已被插入文档中)。update:所在组件的 VNode 更新时调用,但是可能发生在其子 VNode 更新之前。指令的值可能发生了改变,也可能没有。但是你可以通过比较更新前后的值来忽略不必要的模板更新 (详细的钩子函数参数见下)。当数据发生更新时 就会调用。
在使用自定义指令时 如果我们想传递一些参数使用,此时在自定义指令的bind(el,binding)函数中可以进行参数接收
el:指令所绑定的元素,可以用来直接操作 DOM。
binding:一个对象,包含以下 property:
name:指令名,不包括v- 前缀。当有多个单词组成指令名字时,推荐使用‘ - ’ 连接多个单词value:指令的绑定值,例如:v-my-directive="1 + 1" 中,绑定值为2。oldValue:指令绑定的前一个值,仅在update 和componentUpdated 钩子中可用。无论值是否改变都可用。expression:字符串形式的指令表达式。例如v-my-directive="1 + 1" 中,表达式为"1 + 1"。arg:传给指令的参数,可选。例如v-my-directive:foo 中,参数为"foo"。modifiers:一个包含修饰符的对象。例如:v-my-directive.foo.bar 中,修饰符对象为{ foo: true, bar: true }。
对象式自定义指令
在vue实例中添加节点 directives 此节点专门用来定义私有自定义指令:
//在directives属性下 可以定义私有自定义指令 directives:{ //此处的aaa就是自定义指令的名字 aaa:{ //当指令绑定到标签上时 会立即触发 bind事件 bind(el){ el.style.aaa = "green" } } }注意:在上边代码中 aaa就是指令的名字 调用时 要加上v- v-aaa就是此指令的全名 在指令对象中要添加一个bind()函数 只要把指令添加到标签上,那么就可以自动触发 这个bind函数,括号中的el就是触发当前指令的元素DOM对象。 这里是组件01
传递参数的自定义指令使用
//在directives属性下 可以定义私有自定义指令 directives:{ //此处的color就是自定义指令的名字 color:{ //当指令绑定到标签上时 会立即触发 bind事件 //可以直接从binding的value属性中取到 指令的数据 bind(el,binding){ el.style.color = binding.value } } } 这里是组件01
此时存在问题: 如果我们不是通过指令直接赋值,而是通过组件中的data数据来给指令进行赋值,此时,如果我们在指令绑定之后,再次修改了该data数据的值,此时 这个结果不会同步更新到DOM结构中。
使用update()方法 解决这个问题,该方法会在DOM结构更新时调用一次,就可以保证指令中的数据发生更新后 也可以把数据的结果同步更新到DOM结构中。
函数式自定义指令
在很多时候,你可能想在 bind 和 update 时触发相同行为,而不关心其它的钩子。比如这样写:
//全局指令中Vue.directive('color-swatch', function (el, binding) { el.style.backgroundColor = binding.value})//局部指令中此时我们的bind和update函数触发的行为是完全一样的 我们可以把指令对象简写成一个函数使用。
注意:
指令的名字 如果是多个单词 推荐使用 ‘ - ’ 连接。指令中的内置函数中的this指向window对象。
3.4 全局自定义指令
注册一个全局自定义指令 Vue.directive(参数一,参数二)参数一:指的是指令的名字参数二:指的是指令的配置 可以是对象,也可以是函数Vue.directive('focus', { // 当被绑定的元素插入到 DOM 中时…… inserted: function (el) { // 聚焦元素 el.focus() }})
四、项目中axios的相关配置
**问题一:**在以后的vue-cli项目开发中我们很多时候都会发送大量的请求,按照以往的方式 我们会在每一个用到axios的vue文件中导入axios文件,这样比较麻烦 所以我们可以把axios挂载到vue 实例的原型对象上。
main.js文件:
//先在main.js文件中导入axios模块import axios from 'axios'//把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用//this.$ 此时的$= axios
问题二: 重复的导入axios是一个问题,其次每一次发送请求时 都要填写完整的路径也是一个问题,不利于后期的维护,所以我们可以在挂载axios之前 先对axios进行全局的路径配置:
//1.先在main.js文件中导入axios模块import axios from 'axios'//2.挂载之前先配置axios的BaseUrl路径axios.defaults.baseURL = '直接使用//this.$ 此时的$= axios
五、总结与作业
移动端页面头部组件的封装:
s挂载到vue 实例的原型对象上。
main.js文件:
//先在main.js文件中导入axios模块import axios from 'axios'//把axios模块挂载到vue的原型对象上,这样只要是vue的实例对象就都可以使用这个axios模块了,调用的时候 直接使用//this.$ 此时的$= axios
问题二: 重复的导入axios是一个问题,其次每一次发送请求时 都要填写完整的路径也是一个问题,不利于后期的维护,所以我们可以在挂载axios之前 先对axios进行全局的路径配置:
//1.先在main.js文件中导入axios模块import axios from 'axios'//2.挂载之前先配置axios的BaseUrl路径axios.defaults.baseURL = '直接使用//this.$ 此时的$= axios
五、总结与作业
移动端页面头部组件的封装:[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ULLtc3kJ-1655272887374)(assets/image-20220216150337599.png)]
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~